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前 言 (第 3 版 ) 


编写 目的 与 背景 
众所周知 ， 当 前 社会 需求 和 高 校 课程 设置 严重 脱节 ， 一 方面 企业 找 不 到 可 迅速 上 手 的 人 才 ， 另 一 


方面 大 学 生 就 业 难 。 如 果 有 一 些 面向 工作 应 用 的 案例 参考 书 ， 让 大 学 生得 以 参考 ， 并 能 亲手 去 做 ， 势 
必 能 缓解 这 种 矛盾 。 本 书 就 是 这 样 一 本 书 : 项 目 开发 案例 型 的 、 面 向 工作 应 用 的 软件 开发 类 图 书 。 编 


第 四 ， 丛 书 第 1 版 于 2008 年 6 月 出 版 ， 于 2011 年 1 月 改版 ， 因 为 编写 细腻 ， 配 备 全 程 视频 备 受 
读者 瞩目 ， 丛 书 累计 销售 16 万 册 ， 成 为 近年 来 最 受 欢迎 的 软件 开发 项 目 案例 类 从 书 之 一 。 


在 以 上 背景 下 ， 我 们 根据 读者 朋友 的 反馈 ， 与 时 俱 进 ， 对 从 书 进行 了 改版 。 
本 书 特点 


铬 ml 视频 讲解 


对 于 初学 者 来 说 ， 视 频 讲解 是 最 好 的 导师 ， 它 能 够 引导 初学 者 快速 入 门 ， 使 初学 者 感受 到 编程 的 
快乐 和 成 就 感 ， 增 强 进一步 学 习 的 信心 。 鉴 于 此 ， 本 书 为 每 一 个 案例 都 配备 了 视频 讲解 ， 初 学 者 可 以 
通过 视频 讲解 实现 案例 中 的 功能 。 


磋 典 型 案例 


本 书 案例 均 从 实际 应 用 角度 出 发 ， 应 用 了 当前 流行 的 技术 ， 涉 及 的 知识 广泛 ， 读 者 可 以 从 每 个 案 
例 中 积累 丰富 的 实战 经 验 。 


太 代 码 注释 


为 了 便于 读者 阅读 程序 代码 ， 书 中 的 代码 均 提 供 了 详细 的 注释 ， 并 且 整 齐 地 纵向 排列 ， 可 使 读者 
快速 领略 作者 意图 。 
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的 代码 贴 十 

案例 类 书籍 通常 会 包含 大 量 的 程序 代码 ， 宛 长 的 代码 往往 令 初 学 者 望 而 生 旦 。 为 了 方便 读者 阅读 
和 理解 代码 ， 本 书 避 免 了 连续 大 篇 幅 的 代码 ， 将 其 分 割 为 多 个 部 分 ， 并 对 重要 的 变量 、 方 法 和 知识 点 
设计 了 独 具 特 色 的 代码 贴 士 。 


他 知识 扩展 


为 了 增加 读者 的 编程 经 验 和 技巧 ， 书 中 每 个 案例 都 标记 有 注意 、 技 巧 等 提示 信息 ， 并 且 在 每 章 中 
都 提供 有 一 项 专题 技术 。 


本 书 约定 


由 于 篇 幅 有 限 ， 本 书 每 章 并 不 能 逐一 介绍 案例 中 的 各 模块 。 作 者 选择 了 基础 和 典型 的 模块 进行 介 
绍 ， 对 于 功能 重复 的 模块 ， 由 于 技术 、 设 计 思 路 和 实现 过 程 基本 雷同 ， 因 此 没有 在 书 中 体现 。 读 者 在 
学 习 过 程 中 若 有 相关 疑问 ， 请 登录 本 书 官方 网 站 。 本 书 中 涉及 的 功能 模块 在 光盘 中 都 附带 有 视频 录像 ， 
方便 读者 学 习 。 


适合 读者 


本 书 适 合作 为 计算 机 相关 专业 的 大 学 生 、 软 件 开发 相关 求职 者 和 爱好 者 的 毕业 设计 和 项 目 开发 的 
参考 书 。 
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九 九 度 供求 信息 网 


(Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


在 金 球 知 识 经 济 和 信息 化 高 速 发 展 的 今天 ， 信 息 化 是 决定 企业 成 
败 的 关键 因素 ， 企 业 需 要 在 网 站 上 发 布 供求 信息 ， 以 促使 企业 在 同 领 
域 中 得 到 突飞猛进 的 发 展 。 

一 个 广泛 的 、 快 速 的 、 自 由 的 信息 交流 平台 ， 为 用 户 带 来 方便 的 
同时 ， 也 会 给 企业 带 来 无 限 商机 。 于 是 ， 以 互联 网 为 基础 的 信息 交流 
平台 ， 即 九 九 度 供求 信息 网 出 现 了 。 九 九 度 供求 信息 网 致力 于 优化 信 
息 交流 ， 实 现 信 息 的 快速 交流 。 通 过 阅读 本 章 ， 可 以 学 习 到 : 

WI 使 当前 窗口 承载 框架 页 中 的 起 链接 页 面 

MW 如 何 自动 计算 以 系统 日 期 为 基数 的 相对 日 期 

站 do…While 循环 语句 的 应 用 

”查询 关键 字 描 红 枝 术 

WI 在 Windows 操作 系统 下 搭建 PHP 开发 环境 

WI 在 Windows 操作 系统 下 创建 数据 库 和 数据 表 
PowerDesigner 建 模 的 应 用 
单元 测试 技术 
框架 技术 在 Web 网 站 中 的 应 用 
如 何 发 布 网 站 
表单 数据 的 两 种 提交 方式 
应 用 phpMyAdmin 工具 创建 和 删除 数据 库 、 数 据 表 
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Ll 和 开 妆 和 对 


X XX 信息 科技 有 限 公 司 是 一 家 以 整合 渠道 资源 为 主 的 高 科技 公司 。 为 了 扩大 企业 规模 ， 增 强 企 
业 的 竞争 力 , 该 公司 决定 向 多 元 化 发 展 , 借助 Intemet 在 国内 的 快速 发 展 , 聚集 部 分 资金 投入 网 站 建设 ， 
为 企业 和 用 户 提供 综合 信息 服务 ， 以 向 企业 提供 有 偿 信 息 服务 为 盔 利 方式 ， 打 造 一 个 全 新 的 供求 信息 
网 。 例 如 ， 提 供 企业 广告 、 发 布 各 类 免费 供求 信息 、 发 布 企业 付费 信息 等 服务 方式 。 现 需要 委托 其 他 
单位 开发 一 个 综合 信息 网 站 。 


1.2 系统 分 析 


1.2.1 需求 分 析 


对 于 信息 网 站 来 说 ， 用 户 的 访问 量 是 至 关 重要 的 。 如 果 网 站 的 访问 量 很 低 ， 那 么 就 很 少 有 企业 会 
要 求 为 他 提供 有 偿 服 务 ， 也 就 没有 利润 可 言 了 。 因 此 ， 信 息 网 站 必须 为 用 户 提供 大 量 的 、 免 费 的 、 有 
价值 的 信息 才能 够 吸引 用 户 。 为 此 ， 网 站 不 仅 要 为 企业 提供 各 种 有 偿 服 务 ， 还 需要 额外 为 用 户 提供 大 
量 的 无 偿 服 务 。 通 过 与 企业 的 实际 接触 和 沟通 ， 确 定 网 站 应 包括 招聘 信息 、 求 职 信息 、 培 训 信息 、 公 
寓 信 息 、 家 教 信息 、 车 辆 信息 、 物 品 求购 、 物 品 出 售 、 求 兑 出 况 、 寻 求 合 作 、 企 业 广 告 等 服务 。 
通过 实际 调查 ， 要 求 供求 信息 网 具有 以 下 功能 


回 界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵 活 ， 树 立 企业 形象 。 

回 ”实现 强大 的 供求 信息 查询 ， 支 持 模糊 查询 。 

回 用户 不 需要 注册 ， 便 可 免费 发 布 供求 信息 。 

回 ”免费 发 布 的 供求 信息 必须 经 后 台 审 核 后 才能 正式 发 布 ， 避 免 不 良 信息 。 
回 “支持 海量 数据 录入 。 

回 ” 由 于 供求 信息 数据 量 大 ， 后 台 应 该 可 以 随时 清理 数据 。 


1.2.2 可行 性 分 析 


根据 《GB8567 一 88 计算 机 软件 产品 开发 文件 编制 指南 》 中 可 行 性 分 析 的 要 求 ， 制 定 可 行 性 研究 报 
告 如 下 。 


1. 引言 


(1) 编写 目的 

为 了 给 企业 的 决策 层 提 供 是 否 进行 项 目 实施 的 参考 依据 ， 现 以 文件 的 形式 分 析 项 目的 风险 、 项 目 
需要 的 投资 与 效益 。 

(2) 背景 

XXX 信息 科技 有 限 公司 是 一 家 以 整合 渠道 资源 为 主 的 高 科技 公司 。 企 业 为 了 不 断 满足 客户 的 需 
求 ， 为 达到 企业 在 同行 业 领域 中 的 领先 地 位 ， 现 需要 委托 其 他 公司 开发 一 个 综合 信息 网 ， 项 目 名 称 为 
九 九 度 供求 信息 网 。 
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2. 可 行 性 研究 的 前 提 

(1) 要 求 

九 九 度 供求 信息 网 要 求 能 够 提供 信息 搜索 、 信 息 定位 描 红 、 发 布 免费 信息 、 发 布 付费 信息 、 发 布 
企业 广告 和 对 各 类 发 布 的 信息 进行 审核 、 删 除 、 检 索 等 功能 。 

(2) 目标 

九 九 度 供求 信息 网 的 主要 目标 是 提供 强大 的 搜索 功能 、 准 确 的 信息 描 红 定 位 功能 、 付 费 信息 的 管 
理 功 能 、 免 费 信息 的 审核 和 删除 功能 。 

(3) 条 件 、 假 定 和 限制 

项 目 需要 在 两 个 月 内 交付 用 户 使 用 。 系 统 分 析 师 需要 3 天 内 到 位 ， 用 户 需要 4 天 时 间 确 认 需 求 分 
析 文 档 。 去 除 员工 两 个 月 的 正常 休息 日 16 天 ， 那么 程序 开发 人 员 需 要 在 1 个 月 零 几 天 的 时 间 内 进行 系 
统 设计 、 程 序 编码 、 系 统 测 试 、 程 序 调试 和 网 站 部 署 工作 。 

(4) 评价 尺度 

根据 用 户 的 要 求 ， 系 统 应 以 搜索 引擎 为 主 ， 对 于 发 布 的 供求 信息 应 能 及 时 准确 地 保存 、 审 核 、 查 
询 、 描 红 定 位 。 由 于 用 户 存在 多 个 营业 点 ， 系 统 应 具有 局 域 网 操作 的 能 力 ， 在 多 个 营业 点 同时 运行 系 
统 时 ， 系 统 中 各 项 操作 的 延 时 不 能 超过 10 秒 钟 。 此 外 ， 在 系统 出 现 故 障 时 ， 应 能 及 时 进行 恢复 。 


3. 投资 及 效益 分 析 


(1) 支出 

根据 系统 的 规模 及 两 个 月 的 项 目 开发 周期 ， 公 司 决 定投 入 5 个 人 。 因 此 ， 公 司 将 直接 支付 8 万 元 
的 工资 及 各 种 福利 待遇 。 在 项 目 安装 及 调试 阶段 ， 用 户 培训 、 员 工 出 差 等 费用 支出 需要 2 万 元 。 在 项 
目 维护 阶段 预计 需要 投入 2 万 元 的 资金 ， 累 计 项 目 投入 需要 12 万 元 资金 。 

(2) 收益 

用 户 提供 项 目 资金 30 万 元 。 对 于 项 目 运行 后 进行 的 改动 ,采取 协商 的 原则 根据 改动 规模 额外 提供 
资金 。 因 此 从 投资 与 收益 的 效益 比 上 ， 公 司 可 以 获得 18 万 元 的 利润 。 

项 目 完 成 后 ， 将 给 公司 提供 资源 储备 ， 包 括 技术 、 经 验 的 积累 ， 以 后 再 开发 类 似 的 项 目 时 ， 可 以 
极 大 地 缩短 项 目 开 发 周期 。 

4. 结论 


根据 上 面 的 分 析 ， 技 术 上 不 会 存在 问题 ， 因 此 项 目 延 期 的 可 能 性 很 小 ， 在 效益 上 ， 公 司 投入 5 个 
人 、 两 个 月 的 时 间 获 利 18 万 元 ， 比 较 可 观 ， 在 公司 今后 的 发 展 上 ， 可 以 储备 网 站 开发 的 经 验 和 资源 ， 
因此 认为 该 项 目 可 以 开发 。 


1.2.3 ”编写 项 目 计划 书 

根据 《GB8567 一 88 计算 机 软件 产品 开发 文件 编制 指南 》 中 的 项 目 开发 计划 要 求 ， 结 合 单位 实际 情 
况 ， 设 计 项 目 计划 书 如 下 。 

1. 引言 


(1) 编写 目的 
为 了 保证 项 目 开 发 人 员 按 时 保质 地 完成 预订 目标 ， 更 好 地 了 解 项 目 实际 情况 ， 按 照 合 理 的 顺序 开 
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展 工作 ， 现 以 书面 的 形式 将 项 目 开发 生命 周期 中 的 项 目 任务 范围 、 项 目 团队 组 织 结构 、 团 队 成 员 的 工 
作 责 任 、 团 队 内 外 沟通 协作 方式 、 开 发 进度 、 检 查 项 目 工作 等 内 容 描 述 出 来 ， 作 为 项 目 相关 人 员 之 间 
的 共识 和 约定 以 及 项 目 生命 周期 内 的 所 有 项 目 活动 的 行动 基础 。 

(2) 背景 

九 九 度 供求 信息 网 是 本 公司 与 X X X 信 息 科技 有 限 公 司 签 定 的 待 开发 项 目 ， 网 站 性 质 为 信息 服务 
类 型 ， 可 为 信息 发 布 者 有 偿 或 无 偿 提供 招聘 、 求 职 、 培 训 、 求 购 、 公 寓 、 车 辆 、 房 情 和 出 售 等 信息 。 
项 目 周 期 为 两 个 月 ， 项 目 背景 规划 如 表 1.1 所 示 。 


表 1.1 项 目 背景 规划 


项 目 名 称 签 定 项 目 单位 项 目 承担 部 门 
甲 方 ， XXX 信息 科技 有 限 公司 设计 部 门 
九 九 度 供求 信息 网 开发 部 门 
乙方 ，X X X 网 络 科技 有 限 公 司 乙方 : 张 经 理 测试 部 门 


2. 概述 


(1) 项 目 目 标 

项 目 目标 应 当 符合 SMART 原则 ， 把 项 目 要 完成 的 工作 用 清晰 的 语言 描述 出 来 。 九 九 度 供求 信息 
网 的 项 目 目标 如 下 : 

九 九 度 供求 信息 网 主要 用 来 为 用 户 提供 信息 服务 ， 对 于 生活 和 工作 中 的 各 类 信息 都 应 尽 可 能 地 全 
部 包括 在 内 ， 例 如 人 公寓、 求职、 招聘、 培训、 招商、 房屋 、 车 辆 、 出 售 、 求 购 等 信息 。 项 目 发 布 后 ， 
要 实现 能 够 为 用 户 生 活 和 工作 带 来 极 大 的 方便 并 提高 企业 知名 度 、 为 企业 产品 宣传 节约 大 量 成 本 的 目 
标 。 整 个 项 目 需要 在 两 个 月 的 期 限 结束 后 ， 交 给 客户 进行 验收 。 

(2) 产品 目标 与 范围 

一 方面 九 九 度 供求 信息 网 能 够 为 企业 节省 大 量 人 力 资源 ， 企 业 不 再 需要 大 量 的 业务 人 员 去 跑 市 场 ， 
间接 为 企业 节约 了 成 本 。 另 一 方面 ， 九 九 度 供求 信息 网 能 够 收集 海量 供求 信息 ， 将 会 有 大 量 用 户 访问 
网 站 ， 有 助 于 提高 企业 知名 度 。 

(3) 应 交付 成 果 

项 目 开发 完成 后 ， 交 付 的 内 容 如 下 : 

以 光盘 的 形式 交付 九 九 度 供求 信息 网 的 源 程序 、 网 站 数据 库 文件 、 系 统 使 用 说 明 书 。 

客户 方 应 用 自己 的 服务 器 ， 因 此 需要 乙方 架设 Apache 服务 器 、 安 装 PHP 开发 环境 、 协 助 甲 

方 购买 域名 ， 将 开发 的 九 九 度 供求 信息 网 发 布 到 互联 网 上 运行 。 
网 站 发 布 到 互联 网 上 以 后 ， 进 行 后 期 的 6 个 月 无 偿 维护 与 服务 ， 超 过 6 个 月 后 进行 网 站 有 偿 
维护 与 服务 。 

(4) 项 目 验收 方式 与 依据 

项 目 验收 分 为 内 部 验收 和 外 部 验收 两 种 方式 。 在 项 目 开 发 完成 后 ， 首 先进 行内 部 验收 ， 由 系统 测 
试 员 根据 用 户 需 求 和 项 目 目标 进行 验收 。 项 目 在 通过 内 部 验收 后 交 给 用 户 进行 验收 ， 验 收 的 主要 依据 
为 需求 规格 说 明 书 。 

3. 项 目 团队 组 织 


(1) 组 织 结构 
为 了 完成 九 九 度 供求 信息 网 的 项 目 开发 ， 公 司 组建 了 一 个 临时 的 项 目 团队 ， 由 项 目 经 理 、 系 统 分 
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析 师 、PHP 开发 工程 师 、 网 页 设计 师 和 系统 测试 员 构 成 ， 如 图 1.1 所 示 。 


项 目 经 理 


六 ”六 


系统 分 析 师 了 IE 开 发 工程 师 网 页 设计 师 系统 测试 员 
图 1.1 项 目 团队 组 织 结构 图 


(2) 人 员 分 工 
为 了 明确 项 目 团队 中 每 个 人 的 任务 分 工 ， 现 制定 人 员 分 工 表 ， 如 表 1.2 所 示 。 
表 1.2 人 员 分 工 表 
姓 名 工作 描述 
负责 项 目的 审批 、 决策 的 实施 、 项 目的 前 期 分 
张 达 明 | MBA 项 目 开发 部 “| 项 目 经 理 析 、 策 划 、 项 目 开发 进度 的 跟踪 、 项 目 质量 的 


检查 
周 兴 伟 负责 系统 功能 分 析 、 系 统 框架 设计 

邹 紫 下 负责 软件 前 后 台 设 计 与 编码 

王 惠子 负责 网 页 风格 的 确定 、 网 页 图 片 的 设计 
舒心 怡 对 软件 进行 测试 、 编 写 软件 测试 文档 


1.3 系统 设计 


1.3.1 系统 目标 


根据 需求 分 析 的 描述 以 及 与 用 户 的 沟通 ， 现 制定 网 站 实现 如 下 目标 : 

系统 采用 人 机 对 话 方式 ， 界 面 简洁 、 框 架 清晰 、 美 观 大 方 。 

灵活 快速 地 填写 供求 信息 ， 使 信息 传递 更 快捷 。 

信息 查询 灵活 、 方 便 ， 数 据 存储 安全 可 靠 。 

实现 强大 的 后 台 审核 功能 。 

实现 强大 的 搜索 引擎 ， 支 持 模糊 查询 、 关 键 字 描 红 功能 等 。 

对 用 户 输入 的 数据 ， 系 统 进行 严格 的 数据 检验 ， 尽 可 能 排除 人 为 的 错误 。 

网 站 最 大 限度 地 实现 易 维护 性 和 易 操作 性 。 

为 充分 展现 网 站 的 交互 性 ， 供 求 信息 网 采用 动态 网 页 技术 实现 用 户 信息 在 线 发 布 。 
有 具备 完善 的 后 台 管 理 功能 ， 能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 


加 


加 


加 回回 轿 加 加 加 


@ 
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1.3.2 系统 功能 结构 


九 九 度 供求 信息 网 前 台 功能 结构 图 如 图 1.2 所 示 。 


图 1.2 九 九 度 供求 信息 网 前 台 功能 结构 图 
九 九 度 供求 信息 网 后 台 功 能 结构 图 如 图 1.3 所 示 。 


图 1.3 九 九 度 供求 信息 网 后 台 功 能 结构 图 
1.3.3 ”系统 流程 图 
九 九 度 供求 信息 网 的 系统 流程 如 图 1.4 所 示 。 
1.3.4 系统 预览 


九 九 度 供求 信息 网 由 多 个 程序 页 面 组 成 。 下 面 列 出 几 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 

前 台 首 页 如 图 1.5 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 查询 、 企 业 广告 信息 显示 、 后 台 登 录入 口 等 功 
能 。 搜 索引 擎 页 面 如 图 1.6 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 快速 检索 、 查 询 关 键 字 描 红 等 功能 。 

发 布 免费 信息 页 面 如 图 1.7 所 示 , 该 页 面 用 于 实现 发 布 分 类 的 免费 信息 功能 。 付 费 信息 管理 页 面 如 
图 1.8 所 示 ， 该 页 面 用 于 实现 付费 信息 分 类 查看 、 付 费 信息 审核 、 付 费 信息 删除 等 功能 。 


@@ 
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FP 
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了 


rr 
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se 


1.5 前 台 首页 (光盘 \TM\01\99pursey\index.php) ”图 1.6 搜索 引擎 (光盘 \TM\01\99pursey\admin\findinfo.php》 


图 1.7 发 布 免费 信息 (光盘 \TM\01\99pursey\release.php) 


图 1.8 付费 信息 管理 (光盘 \TM\01\99purseyadmin\find fufei.php) 


@ 
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免费 信息 管理 页 面 如 图 1.9 所 示 ， 该 页 面 用 于 实现 免费 信息 分 类 查看 、 免 费 信息 审核 、 免 费 信息 删 
除 等 功能 。 管 理 员 登录 页 面 如 图 1.10 所 示 ， 该 页 面 用 于 实现 对 管理 员 登 录 的 用 户 名 和 密码 进行 验证 等 
功能 。 


Et 
TD 


ET 


图 1.9 免费 信息 管理 (光盘 \TMVWO1\99purseyadminfinddd.php) 图 1.10 管理 员 登 录 (光盘 \TM\01\99pursey\admin\login.php) 
1.3.5 ”开发 环境 


在 开发 九 九 度 供求 信息 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows Server 2003/Linux (推荐 )。 
服务 器 : Apache 2.2.8。 

PHP 软件 : PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.10.3。 
开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辨 率 : 最 佳 效果 为 1024x768 像素 。 


客户 端 
浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辩 率 : 最 佳 效果 为 1024x768 像素 。 


1.3.6 ”文件 夹 组 织 结构 


NP 办 办 办 办 办 办 多 


加 回 


在 编写 代码 之 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 sO er 下 
夹 先 创建 出 来 〈 例 如 ， 创 建 一 个 名 为 admin 的 文件 夹 ， | Te 
用 于 保存 网 站 的 后 台 文 件 ), 这 样 不 但 可 以 方便 以 后 的 $0 
开发 工作 ， 也 可 以 规范 网 站 的 整体 架构 。 笔 者 在 开发 es 


九 九 度 供求 信息 网 时 ， 设 计 了 如 图 1.11 所 示 的 文件 夹 


全 


图 1.11 文件 夹 组 织 结构 
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组 织 结构 图 。 在 开发 时 ， 只 需要 将 所 创建 的 文件 保存 在 相应 的 文件 夹 中 即 可 。 


1.4 在 Windows 操作 系统 下 搭建 PHP 开发 环境 


PHP 能 否 高 效 、 稳 定 地 运行 依赖 于 服务 器 的 编译 和 执行 ， 本 节 主 要 介绍 如 何在 微软 的 Windows 操 
作 系 统 中 架设 安全 、 可 靠 的 PHP 运行 环境 。 


1.4.1 在 Windows 下 应 用 AppServ 快速 配置 PHP 开发 环境 


AppServ 是 PHP 网 页 架 站 工具 组 合 包 , 可 以 将 网 络 上 免费 的 架 站 资源 重新 包装 成 单一 的 安装 程序 。 
它 提 供 了 简易 、 快 速 的 PHP 运行 环境 的 搭建 机 制 ， 读 者 只 需 按照 普通 应 用 软件 的 安装 方式 就 可 以 完成 
Apache+MySQL+PHP+phpMyAdmin 的 安装 与 配置 工作 。 
下 面 以 AppServ-win32-2.5.10 为 例 来 介绍 AppServ 的 安装 和 使 用 方法 。 
安装 AppServ 之 前 应 从 官方 网 站 http://www.appservnetwork.com 下 载 最 新 版 本 的 AppServ-win32- 
2.5.10.exe 安装 程序 。 
[| 视频 录像 : TM\01\Ix\ 在 Windows 下 应 用 AppServ 快速 配置 PHP 开发 环境 .swf 
在 Windows 操作 系统 下 应 用 AppServ 快速 配置 PHP 开发 环境 的 操作 步 又 如 下 : 
(1) 双击 AppServ-win32-2.5.10.exe 文件 ， 打 开 如 图 1.12 所 示 的 AppServ 启动 窗口 。 
(2) 单 击 图 1.12 中 的 Next 按钮 ， 打 开 如 图 1.13 所 示 的 AppServ 安装 协议 窗口 。 


Welcome to the AppServ 2.5.10 


Aorement 
Please review thelcense terms before instaling Appser 2.5.10, 
Setup Wizard 


This wizard mil guice you through the nstalation of AppSery Press Page Down to see the rest of the ageement 
2.5.10, 


OW LS GRAL PULIC LCSE 
要 that you dooe al other spplicatiora Version 2.1, Februan 

before starting Setup. This wil make itpossble th update 

Jevant syaten fles witouthovng to rebeot your op mt oo ge ef 

rer emple 

和 

idkNextto continue, this icense dorument, but changng it s not alowed. 


Veson of be besser Ph, taco co 
了 hence 
] 


诡 : 洒 
ke 
be 
局 
[9 

[未 
O08 
oo 


NMus 


Hf you arcept the terms of he agreenent dickI Agee to continue. You must accept the 
agreement te instal AppSery 2.5.10. 


| 
1.12 AppServ 启动 窗口 图 1.13 AppServ 安装 协议 窗口 

(3) 单 击 图 1.13 中 的 IAgree 按钮 ， 打 开 如 图 1.14 所 示 的 窗口 ， 在 该 窗口 中 可 以 设置 AppServ 的 
安装 路 径 (默认 安装 路 径 一 般 为 C:AppServ)。AppServ 安装 完成 后 ，Apache、MySQL、PHP 都 将 以 子 
目录 的 形式 存储 到 该 目录 下 。 

(4) 单 击 图 1.14 中 的 Next 按钮 ， 打 开 如 图 1.15 所 示 的 窗口 ， 在 该 窗口 中 可 以 选择 要 安装 的 程序 
和 组 件 〈 默 认为 全 选 状态 )。 

(5) 在 图 1.15 中 单 击 Next 按钮 ， 打 开 如 图 1.16 所 示 的 窗口 ， 该 窗口 主要 用 于 设置 Apache 的 端 


口号 。 


(6) 单 击 图 1.16 中 的 Next 按钮 ， 打 开 如 图 1.17 所 示 的 窗口 。 该 窗口 主要 对 MySQL 数据 库 的 


® 
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root 用 户 的 登录 密码 及 字符 集 进 行 设置 , 这 里 将 字符 集 设 置 为 GB2312 Simplified Chinese, 表示 MySQL 
数据 库 的 字符 集 将 采用 简体 中 文 形式 。 


Choose mnstall Location 
Choose the folder in which toinstal AppServ 2.5. 10. 


Selert the components you want to install, cear the components 
youdo not want ioinstal 


Setup wil nstall AppServ 2.5. 10 in the folowing folder, To install in a different folder, cick 
Browse and select another folder, Cick ljexiincaniin 


设置 AppServ 的 安装 路 径 


Destnation Folder 


CiWppserv 


Space required: 56.4ME 
Space avalable; 1.768 


Mpoche HTTP Server nfomation 
Please enter your servers information. 


Server Name (e.g. www.appservnetwork,com) 
Admnistrators Emal Address (e.g. webmaster Ggmal.com) 
winowori echu com 


SN MySQL Server Corfiguration 
MySQL Conigure the MySQL Server instance. 


Please enter Root paseword for MySQ Server 
Enter root password 


Re-enter root password 


pache HTTP Port (Default :80) 


设置 Apache 的 端口 号 


Od Password Support PH MySo APL Rr) 
nabie ImoDe 


Er” 


图 1.16 Apache 端口 号 设置 图 1.17 MySQL 设置 
(7) 单 击 图 1.17 中 的 Install 按钮 后 开始 安装 ， 如 图 1.18 所 示 。 


(8) 安装 完成 后 可 以 选择 在 “开始 ”菜单 的 AppServ 相关 操作 列表 中 启动 Apache 及 MySQL 服 
务 ， 如 图 1.19 所 示 。 


Compleling the AppServ 2.5.10 
Setup Wizard 


appsery 2.5.10hes been nstaled on you comouter. 
Ok Fsh tw cose ths wzard. 


COMPUTES— 


5 


NMuSAL 


em pbse rvre or Ee 


1.18 AppServ 安装 窗口 1.19 AppServ 安装 完成 窗口 
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(9) 安装 好 AppServ 后 ， 整 个 目录 默认 安装 在 C:\AppServ， 此 目录 下 包含 4 个 子 目录 ， 如 图 1.20 
所 示 ， 用 户 可 以 将 所 有 网 页 文件 存放 到 www 目录 下 。 

(10) 打开 浏览 器 ， 在 地 址 栏 中 输入 http://localhost 或 者 http:/127.0.0.1， 如 果 打 开 如 图 1.21 所 示 
的 网 页 ， 则 说 明 AppServ 安装 成 功 。 


The AppServ Open Project - 2.5.10 for Windows 


hdmin Database Manoger Verion ?10 1 
2 PHP miomaton Version 52. 


pe 3 2 rherpo pen sourc shu nstalet packape or Windows mcaes 


Apache 的 存储 目录 


myS9L 的 存储 目录 
PIF 的 存储 路 径 
网 页 文件 及 phpllyAdnin 的 存储 路 径 


图 1.20 AppServ 目录 结构 1.21 AppServ 测试 页 


1.4.2 ” Windows 下 Apache 的 安装 配置 


Cave Longuo0r E SA 


Easy way to bulld Webserver, Database Server with AppServ :-) 


Apache 是 全 世界 使 用 范围 最 广 的 Web 服务 软件 ， 超 过 50% 的 网 站 都 在 使 用 Apache 服务 器 ， 它 以 
高 效 、 稳 定 、 安 全 、 免 费 而 成 为 了 最 受 欢迎 的 服务 器 软件 。 
本 节 主 要 介绍 如 何在 Windows 操作 系统 中 安装 和 配置 Apache 服务 器 。 安 装 Apache 服务 器 前 ， 应 
到 官方 网 站 http:Wwww.apache.org 下 载 Apache 的 安装 程序 。 
谨 M 视 频 录像 : TM\01\Ix\Windows 下 Apache 的 安装 配置 .swf 
在 Windows 操作 系统 下 实现 Apache 的 安装 配置 的 操作 步骤 如 下 : 
(1) 下 载 Apache 的 安装 包 apache_2.0.59-win32-x86-no_ssl.msi 后 ， 双 击 该 安装 包 ， 打 开 如 图 1.22 
所 示 的 Apache 安装 窗口 。 
(2) 在 图 1.22 中 单 击 Next 按钮 ， 打 开 如 图 1.23 所 示 的 Apache 许可 协议 窗口 ， 选 中 I accept the 
terms in the license agreement 单 选 按钮 。 


| 他 Apache MTTP Server 2.0 = Lastal1aU10 划 
Welcome to the Installation Wizard for cea spemment 
Apache HTTP Server 2.0.59 Hease read the folowirg icenss agreement carefuly, 
The Inetalation Weard wil iretal Apache HTTP Server 2.0.59 rr = 
ent eee E| 
httpzmwwrw apache.orgiicenses! 
ERMS AND CONDITIONS FOR USE, RESSoPL mon ho loa 
Dennitons, 选择 接受 许可 协议 
WAN Th Papen epoch Mp booed, “License" shal meal the tems and conditions for use, reproduction, and 
Yomational toate, ietibuion ae definds by Sections 1 through 9 ofthis docurmant 
国 
| Laccept the termns in the hcense agreenent 
a 
wm [EE ee 国 | 
1.22 Apache 安装 窗口 1.23 ”Apache 许可 协议 窗口 
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(3) 在 图 1.23 中 单 击 Next 按钮 ， 打 开 如 图 1.24 所 示 的 HTTP 服务 窗口 。 


(4) 在 图 1.24 中 单 击 Next 按钮 ， 打 开 如 图 1.25 所 示 的 输入 服务 器 信息 窗口 。 输 入 服务 器 的 相关 
信息 ， 如 网 络 域名 、 服 务 器 名 和 管理 员 邮 箱 等 ， 这 里 可 以 根据 用 户 的 实际 情况 输入 。 在 下 方 的 单 选 按 


钮 组 中 ， 如 果 选 择 第 一 项 ， 则 可 以 对 任何 用 户 开放 Apache 服务 ， 同 时 设置 服务 器 的 侦 听 端口 为 80; 
如 果 选 择 第 二 项 ， 则 只 有 本 地 用 户 可 以 连接 和 使 用 Apache 服务 。 


Read This First 


Read this Before Running Apache on Windows. ”4 


Apache HTTP Server 


ie Apache HTTP Server is a powerful and flexible HTTP/.1 compliant web server 
loniginally designed as a replacement for the NCSA HTTP Server, Ithas grown to be 


commercial grade, standards-based sever with freely available source code. 


atestVersion 

Detalls ofthe latest version can be found on the Apache HTTP server prolect page 
lunder 

htpymhttpd apache oral 


5 for Bl Users, on Port 80, 2s a Service ~ Recommended. 
a 到 
Inaalshiald 


Ce eA Po BO Er 


x [2 co | 


<Badk 


图 1.24 HTTP 服务 窗口 1.25 ”输入 服务 器 信息 窗口 


6 注意 如 果 选 择 的 是 第 一 项 ， 又 同时 安装 了 1S， 那 就 必须 修改 IIS 的 默认 端口 ， 否 则 将 导致 


Apache 无 法 正常 工作 。 更改 IIS 的 默认 侦 听 端口 80, 可 以 在 IIS 的 管理 器 中 进行 设置 , 或 者 停止 IIS 
的 服务 也 可 以 。 


(5) 设置 完成 后 ， 单 击 图 1.25 中 的 Next 按钮 ， 打 开 如 图 1.26 所 示 的 选择 安装 方式 窗口 。 这 里 有 


两 种 选择 方式 : Typical 典型 安装 和 Custom 自 定 义 安装 。 通 常 ， 用 户 都 选择 典型 安装 方式 ， 单 击 Next 
按钮 ， 打 开 如 图 1.27 所 示 的 安装 窗口 。 


Setup Type 


KE 2.0 二 Inztsllatia 到 | 
Dectnaton tolder 
Choean th sahp typ that best auts your read 


Mloace select a sotup ba， 


Gchargotondal to dfiernt olar ”fi 
oean pete eb reidked, Peers mdrares 


em sme hel 
poormFieygacn cn [ -ae 
for errrcng ocdes wi ot be Toted ) 
设置 Apache 的 安装 路 径 
ae 


Cam 选择 典型 安装 
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Ta 


| Er ] cm | 和 全 ea | 
图 1.26 选择 安装 方式 窗口 


图 1.27 设置 安装 的 路 径 


(6) 在 如 图 1.27 所 示 的 设置 安装 路 径 窗口 中 可 以 选择 安装 的 路 径 ， 单 击 Change 按钮 ， 在 打开 的 
如 图 1.28 所 示 的 对 话 框 中 修改 文件 的 安装 路 径 。 


7) 在 图 1.28 


h 单 击 OK 按钮 ， 返 回 图 1.27， 单 击 Next 按钮 ， 打 开 如 图 1.29 所 示 的 准备 安装 窗口 。 


(8) 在 图 1.29 中 单 击 Install 按钮 ， 打 开 如 图 1.30 所 示 的 窗口 开始 安装 。 


© 


Ready te mstalthe pregram 
The wimadiready to begn instaloton, 


修改 Apache 的 安装 路 径 


Geenaalto begn the Petalation, 
HF you want bh review orchange any cf youlinstalehion setingx dick Eadk ChdkCancelio 
be wird. 


Ee el le | 
图 1.28 修改 安装 的 路 径 图 1.29 准备 安装 


(9) 图 1.30 中 所 有 文件 复制 完成 后 ，Apache 安装 完成 ， 


打开 如 图 1.31 所 示 的 窗口 ， 单 击 Finish 


按钮 完成 Apache 安装 。 
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图 1.30 复制 文件 图 1.31 安装 完 


(10) 测试 Apache 服务 是 否 安装 成 功 。 选 择 “ 开 始 ”/“ 所 有 程序 ”命令 ， 在 弹出 的 菜单 中 能 够 
看 到 Apache 服务 器 的 相关 操作 列表 ， 同 时 如 果 在 系统 托盘 中 有 一 个 辆 图 标 ， 则 表示 Apache 服务 已 经 
启动 。 

(11) 单 击 短 图 标 后 ,打开 如 图 1.32 所 示 的 窗口 , 在 该 窗口 中 , 选择 Stop 选项 ,表示 停止 Apache 
服务 器 ; 选择 Start 选项 , 表示 启动 Apache 服务 器 (Apache 服务 器 的 当前 状态 为 停止 时 , 该 功能 可 用 ); 
选择 Restart 选项 ， 表 示 重 新 启动 Apache 服务 器 。 双击 短 图 标 后 ， 打 开 如 图 1.33 所 示 的 窗口 ， 可 以 进 
行 启动 和 停止 服务 器 的 各 种 操作 。 
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图 1.33 Apache 服务 器 的 监视 器 


图 1.32 左 键 菜单 
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(12) 在 浏览 器 中 输入 http://localhost/ 或 者 输入 http://127.0.0.1/， 如 果 能 够 浏览 到 如 图 1.34 所 示 
窗口 ， 说 明 Apache 服务 器 安装 成 功 。 

(13) Apache 服务 器 安装 成 功 后 ， 接 下 来 对 。 [sse seeses se 
Apache 服务 器 进行 配置 ， 以 便 Apache 服务 器 能 够 识 上 ed EE 
别 PHP 文件 。 配 置 Apache 服务 器 主要 是 在 Apache 
安装 目录 下 的 conf 子 目录 中 的 httpd.conf 文件 中 进行 这 不 是 你 想 看 见 的 页 面 吧 ? 

的 , 找到 该 文件 并 用 记事 本 等 文本 编辑 器 打开 该 文件 。 | 是 3 人 各 下 

(14) 定位 到 DocumentRoot 一 行 ， 可 以 将 该 路 rr 
径 修改 为 合适 的 路 径 ， 例 如 ， 设 置 为 DocumentRoot” | Seon seanareaar omnes 
Ci\Apache\Apache2\webpage"， 表 示 Apache 服务 器 的 于 ne | 
虚拟 目录 为 Ci\Apache\Apache2\webpage。 

(15) 将 光标 定位 到 DirectoryIndex index.html 
index.html.var， 在 其 后 面 添加 一 个 PHP 默认 页 ， 通 常 是 index.php， 表 示 当 访问 该 服务 器 时 如 果 未 指定 
要 访问 的 PHP 文件 ， 则 默认 访问 index.php 文件 。 更 改 后 的 代码 如 下 : 

DirectoryIndex index.html index.html.var index.php 
(16) 为 了 使 Apache 识别 PHP 的 扩展 名 ， 搜 索 并 定位 到 httpd.conf 文件 的 如 下 部 分 : 
<Directory "C:/Apache/Apache2/cgi-bin"> 
AllowOverride None 
Options None 
Order allow,deny 
Allow from all 
</Directory> 
在 后 面 添加 如 下 两 行 代码 : 


AddType application/x-httpd-php .php .phtml .php3 .php4 
AddType application/x-httpd-php-source .phps 


(17) 为 了 能 够 使 用 模块 功能 ， 模 块 通常 以 DSO 方式 构建 ， 读 者 需要 定位 到 如 下 代码 : 
#LoadModule ssl_module modules/mod_ssl.so 


使 得 能 够 在 使 用 前 获得 指令 的 功能 ， 然 后 以 module 方式 加 载 PHP， 指 向 PHP 5.0 目录 下 的 
php5apache2.dll 文件 ， 添 加 如 下 代码 : 


LoadModule php5_module c:\php5\php5apache2.dll 
1.4.3 ”Windows 下 MySQL 的 安装 配置 
PHP 能 够 支持 市 面 上 的 大 多 数 数据 库 ， 如 MySQL、Access、SQL Server、Oracle 等 ， 而 MySQL 


数据 库 是 PHP 程序 开发 人 员 公 认 的 黄金 搭档 ， 这 不 仅 因为 MySQL 是 完全 网 络 化 的 跨 平 台 关系 型 数据 
库 系 统 ， 而 且 它 也 是 具有 客户 机 /服务 器 体系 结构 的 分 布 式 数据 库 管 理 系 统 。 它 具有 功能 性 强 、 使 用 简 


还 


1.34 测试 Apache 服务 器 是 否 安装 成 功 
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洁 、 管 理 方便 、 运 行 速度 快 、 版 本 升级 快 、 安 全 性 高 等 优点 ， 而 且 MySQL 数据 库 完 全 免费 ， 从 官方 
网 站 http:/www.mysql.com 即 可 免费 下 载 到 最 新 版 本 的 MySQL 安装 包 。 

人 1 视频 录像 : TM\01\Ix\Windows 下 MySQL 的 安装 配置 .swf 

在 Windows 操作 系统 下 实现 MySQL 的 安装 配置 分 为 以 下 几 种 方式 : 


如 果 下 载 的 是 Windows Essentials (x86)， 


例如 mysql-4.1.11-essential-win.exe， 直 接 双击 该 文 


件 安装 即 可 ,安装 完毕 后 , 在 “开始 ”菜单 中 选择 “运行 ”命令 , 在 弹出 的 对 话 框 中 输入 cmd， 


进入 “命令 提示 符 ” 窗 口 ， 输 入 mysqld-ntexe， 即 可 启动 MySQL 服务 ; 如 果 还 想 详细 


也 配置 


MySQL， 则 需要 进入 C:\mysql\bin 目录 , 运行 MySQLInstanceConfig.exe， 并 按 提示 操作 即 可 。 


如 果 下 载 的 是 Windows (x86)， 例 如 mysql-essential-5.0.24-win32.msi， 解 压 该 文件 后 双击 


setup.exe 安装 程序 ， 即 可 逐步 完成 MySQL 的 安装 。 


如 果 下 载 的 是 Without noinstall (unzip in C:\), 直接 将 该 文件 解压 到 指定 的 安装 目录 下 。 在 “ 开 


始 ” 菜 单 中 选择 “运行 ”命令 ， 在 弹出 的 对 话 框 中 输入 cmd， 然 后 进入 “命令 提示 符 ” 窗 口 ， 
输入 mysqld-nt.exe 命令 ， 即 可 启动 MySQL 服务 器 。 


1.4.4 Windows 下 PHP 的 安装 配置 


架设 基于 PHP 的 Web 服务 器 , 安装 PHP 是 必需 的 。 由 于 PHP 的 代码 公开 ,所 以 其 升级 速度 较 快 。 
安装 PHP 之 前 应 从 官方 网 站 http://www.php.net/ 下 载 最 新 版 本 的 PHP 安装 程序 ， 下 面 以 


php-5.1.4-Win32.zip 为 例 讲解 PHP 的 配置 方法 。 


句 1 视 频 有 录像:， TM\01\Ix\Windows 下 PHP 的 安装 配置 .swf 
在 Windows 操作 系统 下 实现 PHP 的 安装 配置 的 操作 步骤 如 下 : 


(1) 成 功 下 载 到 PHP 的 安装 包 后 , 首先 应 对 
其 进行 解压 ， 如 解压 到 Ci\php5 目录 下 ， 如 图 1.35 
所 示 。 

(2) 将 Ci\php5\ 目 录 下 的 所 有 .dll 文件 复制 
到 C:\Windows\system32\ (如果 是 Windows 2000 
操作 系统 ， 则 为 C\WINNT\system32\) 目录 下 。 

(3) 将 Ci\php5\ 目 录 下 的 php.ini-dist 文件 
复制 到 C:\Windows\( 如果 是 Windows 2000 操作 
系统 ， 则 为 CA\WINNT\) 目录 下 ， 然 后 将 
php.ini-dist 重 命名 为 php.ini, 并 用 记事 本 打开 该 
文件 进行 编辑 ， 具 体 编辑 方式 如 下 : 

首先 ， 定 位 到 该 文件 register_globals = Of 
所 在 行 ， 将 O 任 改 为 On。 代码 如 下 : 


register_globals = On 
其 次 ， 定 位 到 该 文件 extension_dir=".\" 所 在 行 ， 


extension_dir="C:\phpS\ext" 
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图 1.35 PHP 解压 后 的 文件 


将 路 径 改 为 "Ci:\php5\ext"。 代 码 如 下 : 
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最 后 ， 查 找 定位 Windows Extensions， 将 下 面 两 行 代 码 : 


; extension=php_gd2.dll 
; extension=php_mysql.dll 


前 面 的 注释 〈 即 分 号 标识 符 ) 去 掉 ， 这 样 PHP 就 可 以 支持 GD2 函数 库 和 MySQL 相关 函数 ， 如 果 想 让 
PHP 支持 其 他 功能 函数 ， 同 样 需 要 将 与 这 些 操作 函数 相关 的 dll 文件 前 的 注释 去 掉 。 


和 0 注意 以 上 为 PHP 的 常规 配置 ， 在 PHP 的 官方 论坛 上 可 以 查 到 更 多 的 详细 配置 由 于 PHP 的 
版 本 有 多 种 ， 所 以 php.ini 的 配置 也 存在 差别 ， 具 体 使 用 时 应 根据 实际 情况 配置 php.ini 文 件 。 


(4) PHP 配置 完成 以 后 ， 需 要 重新 启动 Apache 服务 器 ， 然 后 编写 test.php 文件 并 输入 如 下 代码 : 
<?php 

phpinfo(); 
?> 


将 文件 保存 到 Apache 安装 目录 下 的 webpage 子 目 录 下 ， 然 后 在 浏览 器 中 输入 http://127.0.0.1/ 
testphp， 如 果 显 示 如 图 1.36 所 示 的 页 面 ， 则 说 明 PHP 配置 成 功 。 
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图 1.36 测试 成 功 
- 声 


= 

Ws a 
带 来 不 便 , 选择 D:\ust\local\php 的 目录 时 要 和 在 Linux 下 的 安装 目录 相 匹 配 。 建议 最 好 不 要 选择 中 
间 有 空格 的 目录 ， 如 E:\Program Files\PHP， 这 样 做 会 导致 一 些 未 知 错误 或 崩 演 发 生 。 


1.4.5 在 Windows 下 设置 IIS+PHP 的 执行 环境 
PHP 可 以 在 很 多 服务 器 下 运行 ， 下 面 介 绍 IIS 的 安装 方法 以 及 PHP 是 如 何在 IIS 服务 器 下 运行 的 。 
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1. llS 的 安装 


Internet 信息 服务 器 (Internet Information Server， 缩 写 为 IS) 是 Microsoft 的 Web 服务 器 ， 它 集成 
于 Windows NT Server 之 中 ， 方 便 易 用 ， 为 Web 应 用 程序 提供 了 功能 强大 的 运行 平台 。 

IIS 目前 最 新 版 本 为 6.0, 在 Windows 2003 Server 操 作 系统 中 默认 安装 ,如 果 没 有 安装 Windows 2003 
Server， 可 以 选择 IIS 5.0， 它 可 以 在 Windows XP 或 者 Windows 2000 操作 系统 中 安装 ， 对 于 开发 而 言 ， 
IIS 的 各 个 版 本 之 间 的 区 别 并 不 大 。 

(1) 在 Windows XP 或 者 Windows 2000 操作 系统 下 安装 IIS 
颁 4 视 频 录 像 : TM\01\Ix\ 在 Windows XP 或 者 Windows 2000 操作 系统 下 安装 [IS.swf 

选择 “开始 ”/“ 设 置 ”/“ 控 制 面板 ”命令 ， 启 动 “ 添 加 /删除 程序 ”应 用 程序 ， 单 击 “ 添 加 /删除 
Windows 组 件 ” 图 标 ， 在 弹出 的 “Windows 组 件 向 导 ” 对 话 框 的 “组 件 ” 列 表 框 中 选择 “Internet 信息 
服务 (IIS)” 选 项 ， 最 后 单 击 “ 下 一 步 ” 按 钮 ， 即 可 完成 IIS 的 安装 。 

(2) 在 Windows 2003 Server 服务 器 下 安装 IIS 

印 和 视频 录像 : TM\01\Ix\ 在 Windows 2003 Server 服务 器 下 安装 IIS.swf 

首先 选择 “开始 ”/“ 控 制 面板 ”命令 ， 然 后 启动 “添加 /删除 程序 ”应 用 程序 ， 单 击 “ 添 加 /删除 
Windows 组 件 ” 图 标 ， 在 弹出 的 “Windows 组 件 向 导 ” 对 话 框 的 “组 件 ” 列 表 框 中 选择 “应 用 程序 服 
务 器 ”选项 ， 然 后 单 击 “ 详 细 人 信息” 按钮 ， 选 择 “Internet 信息 服务 (IIS)” 选 项 ， 最 后 单 击 “ 下 一 步 ” 
按钮 ， 即 可 完成 IIS 的 安装 。 


人 
取 、 运 行 脚本 (如 ASP)、 执 行 (如 ISAPI 应 用 程序 或 CGI ) 选项 。 


2. 架设 PHP 到 IIS 服务 器 


在 架设 PHP 到 IIS 服务 器 之 前 , 首先 应 确保 PHP 已 经 正确 安装 到 系统 中 。 本 节 将 以 Windows 2003 
Server 服务 器 中 的 IIS 6.0 为 例 ， 介 绍 架 设 PHP 到 IIS 服务 器 的 方法 。 

鳃 中 视频 录像 ，TMNOTNIX\ 在 Windows 下 设置 IIS+PHP 的 执行 环境 .swf 

(1) 选择 “开始 ”/“ 管 理工 具 ”/“Intemet 信息 服务 (IIS) 管理 器 ”命令 ， 打 开 如 图 1.37 所 示 
的 mternet 信息 服务 窗口 。 

(2) 右 击 “ 默 认 网 站 ”节点 ， 在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ， 将 打开 如 图 1.38 所 示 的 对 
话 框 。 

(3) 单 击 图 1.38 中 “ 主 目录 ”选项 卡 中 的 “配置 ”按钮 ， 打 开 如 图 1.39 所 示 的 对 话 框 。 

(4) 单 击 图 1.39 中 的 “添加 ”按钮 ， 将 打开 如 图 1.40 所 示 的 对 话 框 ， 单 击 “ 可 执行 文件 ”文本 
框 后 的 “浏览 ”按钮 ， 在 打开 的 窗口 中 选择 php5 目录 下 的 php5isapi.dll 文件 ， 在 “扩展 名 ”文本 框 中 
输入 “.php”， 然 后 单 击 “ 确 定 ” 按 钮 ， 实 现 扩 展 名 的 映射 。 

(5) 单 击 “ 主 目录 ”选项 卡 中 的 “浏览 ”按钮 ， 选 择 网 站 的 路 径 ， 如 图 1.41 所 示 。 

(6) 在 “Internet 信息 服务 (HS) 管理 器 ”窗口 中 单 击 “Web 服务 扩展 ”节点 ， 在 “Web 服务 扩 
展 ” 区 域 中 选择 “所 有 未 知 ISAPI 扩展 ”选项 ， 然 后 单 击 “ 允许” 按钮， 将 该 服务 的 状况 由 “禁止 ” 


设置 为 “允许 ” 如 图 1.42 所 示 。 
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图 1.39 应 用 程序 配置 
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图 1.40 添加 /编辑 应 用 程序 扩展 名 映射 
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图 1.41 设置 网 站 的 路 径 


1.42 ”设置 Web 服务 扩展 


(7) 打开 IS 服务 器 ， 浏 览 index.php 文件 ， 如 果 在 浏览 器 中 打开 网 页 文件 ， 则 说 明 IIS+PHP 运 
行 环境 配置 成 功 ， 如 图 1.43 所 示 。 


@ 


I 


第 1 章 ， 九 九 度 供求 信息 网 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) ”毛毛 所 


PE 
| 


ED bl 
一 一 


图 1.43 IIS+PHP 运行 环境 配置 成 功 
1.5 数据 库 设 计 


1.5.1 数据 库 分 析 


本 系统 是 一 个 中 小 型 的 供求 信息 平台 ， 但 是 由 于 平台 会 涉及 海量 数据 ， 因 此 需要 充分 考虑 到 成 本 问 
题 及 用 于 需求 《如 跨 平 台 ) 等 问题 。 而 MySQL 是 世界 上 最 为 流行 的 开放 源码 的 数据 库 ， 是 完全 网 络 化 
的 、 跨 平台 的 关系 型 数据 库 系统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 


1.5.2 ”数据 库 概念 设计 

根据 前 面 对 系统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实体 分 别 为 免费 信息 
实体 、 付 费 信 息 实体 、 广 告 信息 实体 和 管理 员 实 体 。 下 面 分 别 介绍 这 儿 个 实体 的 E-R 图 。 

1. 免费 信息 实体 

免费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 审 核 状 态 和 发 布 
时 间 属 性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 审核 ,“1” 表 示 “ 是 ”“0” 表 示 “ 和 否 ”。 免 费 信息 实 
体 的 E-R 图 如 图 1.44 所 示 。 

2. 付费 信息 实体 


付费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 发 布 时 间 、 和 截止 
时 间 和 审核 状态 属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 付费 ,“1” 表 示 “ 是 ”%“0” 表 示 “ 否 ” 
付费 信息 实体 的 E-R 图 如 图 1.45 所 示 。 


3. 广告 信息 实体 

广告 信息 实体 包括 编号 、 信 息 标题 、 信 息 内 容 、 发 布 时 间 和 推荐 状态 属性 。 其 中 推荐 状态 属性 用 来 
标识 信息 是 否 在 前 台 显示 ,“1” 表 示 “ 是 ”“0” 表 示 “ 和 否 ” 广告 信息 实体 的 E-R 图 如 图 1.46 所 示 。 

4. 管理 员 实体 

管理 员 实体 包括 编号 、 管 理 员 名 和 加 密 密 码 属性 。 管 理 员 实 体 的 E-R 图 如 图 1.47 所 示 。 
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图 1.46 广告 信息 实体 E-R 图 图 1.47 管理 员 实体 E-R 图 
1.5.3 ”使 用 PowerDesigner 建 模 


在 数据 库 概 念 设 计 中 已 经 分 析 了 本 系统 中 主要 的 数据 实体 对 象 ， 通 过 这 些 实体 可 以 得 出 数据 表 结 

构 的 基本 模型 ， 最 终 实 施 到 数据 库 中 ， 形 成 完整 的 数据 结构 。 下 面 将 介绍 使 用 PowerDesigner 工具 完成 

(1) 运行 PowerDesigner， 并 在 PowerDesigner 主 窗口 中 选择 File/New 命令 ， 在 打开 的 New 对 话 

框 中 选择 Physical Data Model (物理 数据 模型 ，PDM) 列表 项 ， 单 击 OK 按钮 ， 打 开 如 图 1.48 所 示 的 
Choose DBMS 〈 选 择 数据 库 管理 系统 ) 对 话 框 。 站 


(2) 在 DBMS 下 拉 列 表 框 中 选择 MySQL 数据 库 , 这 [ES 


里 选择 MySQL 3.23 版 本 ， 这 是 由 于 当前 使 用 的 6 hore Une the shered DOMS dofraion 

PowerDesigner 的 版 本 还 不 支持 MySQL 5.0， 但 是 随 着 Comy Creake a copy of ihe DBMS defraion in model 

PowerDesigner 版 本 的 升级 将 支持 更 多 的 DBMS。 ED rvsme 
(3) 其 他 选项 采用 默认 设置 即 可 ， 单 击 OK 按钮 ， 打 ET 

开 新 建 的 PDM 窗口 。 在 该 窗口 的 上 方 为 空 的 图 形 窗口 ， - 

下 方 为 输出 窗口 。 其 中 图 形 窗口 的 右 侧 有 一 个 工具 面板 ， 人 


如 图 1.49 所 示 。 
(4) 在 图 1.49 中 单 击 “ 建 立 表 ” 图 标 , 这 时 鼠标 指针 将 显示 为 日 ， 在 图 形 窗口 的 合适 位 置 单 击 鼠 
标 左 键 ， 此 时 在 图 形 窗 口中 将 显示 如 图 1.50 所 示 的 表 符号 。 
Eco 


Table_1 


1.49 工具 面板 图 1.50 表 符号 
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人 如 果 再 单 击 鼠 标 左 键 还 将 出 现 类 似 图 
1.50 所 示 的 表 符号 ， 如 果 想 取消 该 指针 ， 可 以 单 击 工具 面板 中 的 指针 图 标 必 。 


(5) 在 图 1.50 所 示 的 表 符号 上 双击 鼠标 左 键 ， 将 打开 Table Properties ( 表 属性 ) 对话 框 ， 默认 情 
况 下 选择 的 是 General 选项 卡 ， 在 该 选项 卡 的 Name 文本 框 中 输入 表 的 名 字 tb_admin， 此 时 在 Code 文 
本 框 中 也 将 自动 显示 tb_admin， 其 他 选项 选择 默认 即 可 。 

(6) 选择 Columns 选项 卡 ， 首 先 单 击 列 输入 列表 的 第 一 行 ， 将 自动 填写 一 行 信 息 ， 然 后 将 Name 
列 修改 为 ii， 同时 Code 列 也 将 自动 显示 为 id， 再 在 Data Type 列 中 选择 integer unsigned 列表 项 ， 最 后 
选中 P 列 的 复 选 框 ， 此 时 M 列 的 复 选 框 也 将 自动 选中 。 

(7) 按照 步骤 (6) 的 方法 再 添加 两 个 列 name 和 pwd， 如 图 1.51 所 示 。 

(8) 在 图 1.51 中 单 击 “ 应 用 ”按钮 后 ， 双 击 id 列 的 内 容 ， 将 打开 Column Properties 〈 列 属性 ) 
对 话 框 ， 如 图 1.52 所 示 。 默 认 选 择 General 选项 卡 ， 在 该 选项 卡 中 选中 Identity 复 选 框 ， 此 项 操作 用 于 
设置 id 列 为 自动 编号 列 。 


Properties -id (id) 上 加 区 
| Bules | Dependenceies | Yersion Info 
ot Seneral | Datsil | Standard Checks | Additiont Checks 
Nappine | Wotes | Bules | Dependencies | Extenled Deperdencies | Tersion Info Ps 和 
General Colunns | Indexes | Keys | chesk | Seript | Options | Previer a Er | 
思 |' 轨 加 风 | 名曲 氏 XX| 的 态 坏 Comrent | 可 
[TNnel ce | yp ul 
3 ; Ey F Tabe 加 
Daaype fr ?pisployed 
lero 站 pecao [ 
Domar ET 习 同 
PRimykey TF Eocmkey 区 Mandion 
目 ” CE ww | nw | ww | 
图 1.51 ”Columns 选项 卡 图 1.52 Column Properties 对 话 框 


(9) 单 击 “ 应 用 ”按钮 后 ， 再 单 击 “ 确 定 ” 按 钮 ， 关 闭 Column Properties 对 话 框 。 
(10) 单 击 “ 确 定 ” 按 钮 ， 关 闭 Table Properties 对 话 框 ， 完 成 tb_admin 表 的 创建 。 
(11) 按照 步骤 (4) 一 〈10) 的 方法 创建 本 系统 中 的 其 他 数据 表 。 创 建 完成 的 模型 图 如 图 1.53 所 示 。 


i th leaguerinfo 
hin 
= 四 nt ES 
| etining id pe varchar (20) 
区 加 cp ||type varchar (30) titl archar(50) 
i int Spk> ||tite .varchar (100) title varchar(50) content 。 varchar(500) 
name varchar (50) content varchar (500) content varchar(500) linkman varchar(20) 
pmd varchar (50) fdate 。 datetime linkaan 。 varchar(20) tel varchar(30) 
flag 。 int(D) tel varchar (30) sdate 
heckstate int (1) howday date 
edatetime checkstate int (1) 


图 1.53” 九 九 度 供求 信息 网 的 模型 图 


| 


| 
文字 的 格式 ， 可 以 选中 全 部 表 符号 ， 按 CtrlHT 键 ， 在 打开 的 Symbol Format 对 话 框 中 选择 Font 选 
项 卡 ， 在 该 选项 卡 中 设置 相关 内 容 的 字体 及 样式 和 字号 等 。 
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(12) 选择 PowerDesigner/Database/Generate Database 命令 ， 将 打开 Database Generation 对 话 框 ， 
在 该 对 话 框 中 设置 导出 的 脚本 文件 的 名 称 〈 如 pursey.sql) 及 保存 路 径 〈 如 F\ pursey)， 在 下 方 的 各 个 
选项 卡 中 设置 相关 的 导出 内 容 ， 设 置 完毕 后 ， 单 击 “确定 ”按钮 ， 导 出 数据 库 脚本 文件 完成 。 


说 明 通过 以 上 方法 导出 的 脚本 文件 ， 可 以 在 MySQL 的 客户 端 命令 行 窗口 中 通过 source 命令 
执行 ， 并 创建 相应 的 数据 表 。 


1.5.4 创建 数据 库 及 数据 表 


结合 实际 情况 及 对 用 户 需 求 的 分 析 , 可 知 九 九 度 供求 信息 网 中 应 用 的 db_pursey 数据 库 主 要 包含 如 
下 4 个 数据 表 ， 如 图 1.54 所 示 。 


各 数据 表 的 表 结 构 如 图 1.55 一 图 1.58 所 示 。 ed 证 庆 下 
1，tb_admin (管理 员 信息 表 ) tb_admin MYISAM gb2312_chinese_ci 。 管理 员 值 息 表 


也 _advertising MyYISAM 。 gb2312_chinese_cl 。 全 业 三 洁 信 息 表 
th_info MYISAM 。 9b2312_chinese_cl 。 免费 供求 信息 表 


管理 员 信 息 表 主 要 用 于 存储 管理 员 的 信息 。 该 数据 。 | wmagoenie WisAW 962312_ehinasezel 有 并 局 家 
表 的 结构 如 图 1.55 所 示 。 
2. tb_advertising (企业 广告 信息 表 ) 


企业 广告 信息 表 主 要 用 于 存储 企业 发 布 的 广告 信息 。 该 数据 表 的 结构 如 图 1.56 所 示 。 


图 1.54 九 九 度 供求 信息 网 数据 表 


胃 服务 器 : localhost ， 电 数据 库 ; db_pursey， 四 表 : tb_advertising 
FR 整理 Nul 默认 额外 


说 明 
盟 服务 器 : localhost 》 怕 数据 库 : db_pursey》 加 表 :tb_admin 地 再 auto_increment 。 自动 机 号 
字 及 类型 整理 Na 多 认 往外 说明 ‘title varchar(100) gb2312_chinese_cl 否 广告 标题 
i nt4) 理 auto_increment ”自动 编号 content varcha(500) gb2312_chinese_cl 否 广告 内 容 
name varchar(50》 gb2912_chinese_cl 是 AULL 和 营 理 只 扣 fdate datetime 否 发 和 时 间 
Tlag 


pwd varchar(50) gb2312_chinese_cl 是 WULL 管理 民 密 码 


nt) 否 0 推荐 状态 


图 1.55 管理 员 信息 表 结 构 图 1.56 企业 广告 信息 表 结构 
3. tb_info (免费 供求 信息 表 ) 
免费 供求 信息 表 主 要 用 于 存储 用 户 免费 发 布 的 供求 信息 。 该 数据 表 的 结构 如 图 1.57 所 示 。 
4. tb_leaguerinfo (付费 供求 信息 表 ) 
付费 供求 信息 表 主 要 用 于 存储 付费 的 供求 信息 。 该 数据 表 的 结构 如 图 1.58 所 示 。 


图 服务 器 : localhost ， 曲 数据 库 : db_pursey } 图 表 :tb_info 胃 服务 器 : lecalhost ， 局 获 据 库 : db_pursey } 图 表 :tb_leaguerinfo 
字 县 于 Mul 睦 认 术 外 Li] 
Int 人 理 auto_increment 再 a_ncfement 。 自动 本 时 
varcharG0) gb2312_cninese_ci 
varchar(s0) gb2312_chinese_c 


gb2312_chiness_oi 否 信息 交错 
ob2312_chineco_eil 否 生息 行 是 
gb2312_chiress_d 百 二 自 具 容 
gbz312_chinese xl 百 联系 人 
gh2312_chinese_di 否 隘 系 志 话 
发 布 日 其 
其 止 日 类 
审核 者 


图 1.57 免费 供求 信息 表 结构 图 1.58 付费 供求 信息 表 结构 


要 
否 
varchar600) gb2312_chinese_ci 否 
varchar20) gb2312_chinese_ri 于 
varcharG0) -gb2312_chinese_ci 否 
ke nt) 香 
datetime 理 


如 
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1.6 单元 测试 


在 现代 软件 开发 过 程 中 ， 测 试 不 再 作为 一 个 独立 的 生命 周期 ， 而 是 成 为 与 编写 代码 同步 进行 的 开发 
活动 。 单 元 测试 能 够 提高 程序 员 对 程序 的 信心 ， 保 证 程序 的 质量 ， 加 快 软件 开发 速度 ， 使 程序 易于 维护 。 


1.6.1 单元 测试 概述 


在 程序 设计 过 程 中 会 有 多 种 测试 ， 单 元 测试 只 是 其 中 的 一 种 。 单 元 测试 并 不 能 保证 程序 是 完美 无 
缺 的 ， 但 是 在 所 有 的 测试 中 ， 单 元 测试 是 第 一 个 环节 ， 也 是 最 重要 的 一 个 环节 。 单 元 测试 是 一 种 由 程 
序 员 自 行 测试 的 工作 。 简 单 地 说 ， 单 元 测试 就 是 测试 代码 撰写 者 依据 其 所 设想 的 方式 执行 是 否 产生 了 
预期 的 结果 

单元 测试 不 仅 是 无 错 编码 的 一 种 辅助 手段 ， 还 必须 是 可 重复 的 ， 即 无 论 是 在 软件 修改 或 是 移植 到 
新 的 运行 环境 的 过 程 中 都 可 以 进行 单元 测试 。 

与 单元 测试 有 密切 关联 的 开发 活动 包括 代码 走读 (Code review)、 静 态 分 析 (Static analysis ) 和 动 
态 分 析 (Dynamic analysis)。 代 码 走读 就 是 对 软件 的 源 代码 进行 宏观 阅读 ， 整 理 开 发 思路 。 静 态 分 析 就 
是 对 软件 的 源 代 码 进行 研读 ， 查 找 错 误 或 收集 一 些 度量 数据 ， 并 不 需要 对 代码 进行 编译 和 执行 。 动 态 
分 析 就 是 通过 观察 软件 运行 时 的 动作 ， 来 提供 执行 跟踪 、 时 间 分 析 以 及 测试 覆盖 度 方面 的 信息 。 


1.6.2 单元 测试 的 优点 


单元 测试 具有 以 下 优点 : 

回 ”一 种 验证 行为 。 程 序 中 的 每 一 项 功能 都 是 用 测试 来 验证 它 的 正确 性 ， 为 以 后 的 开发 提供 支援 。 
就 算是 开发 后 期 ， 也 可 以 轻松 地 增加 功能 或 更 改 程序 结构 ， 而 不 用 担心 这 个 过 程 中 会 破坏 重 
要 的 东西 ， 而 且 它 为 代码 的 重 构 提供 了 保障 。 这 样 ， 我 们 就 可 以 更 自由 地 对 程序 进行 改进 。 

回 一 种 设计 行为 。 编 写 单元 测试 将 使 用 户 从 调用 者 观察 、 思 考 。 特 别 是 先 写 测试 (test-first)， 
迫使 设计 者 把 程序 设计 成 易于 调用 和 可 测试 的 ， 即 解除 软件 中 的 耦合 。 

加 一 种 编写 文档 的 行为 。 单 元 测试 是 一 种 无 价 的 文档 ， 它 是 展示 函数 或 类 如 何 使 用 的 最 佳 文档 。 
这 份 文档 是 可 编译 、 可 运行 的 ， 并 且 它 保持 最 新 ， 永 远 与 代码 同步 。 

回 具有 回归 性 。 自 动 化 的 单元 测试 避免 了 代码 出 现 回 归 ， 编 写 完成 之 后 ， 可 以 随时 随地 地 快速 
运行 测试 。 


1.7 前 台 首页 设计 


1.7.1 前 台 首 页 概述 


网 站 主页 是 关于 网 站 的 建设 及 形象 宣传 ， 它 对 网 站 生存 和 发 展 起 着 非常 重要 的 作用 ， 应 该 是 一 个 
信息 含量 较 高 、 内 容 较 丰 富 的 宣传 平台 。 九 九 度 供求 信息 网 前 台 首 页 主要 包含 以 下 内 容 : 
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网 站 菜单 导航 《包括 公寓 信息 、 招 聘 信 息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信息 、 车 
辆 信息 、 求 购 信 息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启 示 等 )。 

发 布 免费 的 供求 信息 〈 包 括 公 寓 信 息 、 招 聘 信息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信 
息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启示 等 )。 

推荐 供求 信息 显示 〈 包 括 公寓 信息 、 招 聘 信 息 、 求 职 信息 、 培 训 信息 、 家 教 信 息 、 房 屋 信息 、 
车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启示 等 )， 其 中 ， 付 费 信 息 按时 间 顺 序 
降序 排列 ， 免 费 信息 按时 间 顺 序 分 页 显示 。 

回 ”显示 推荐 的 企业 广告 信息 。 

供求 信息 快速 检索 ， 支 持 模糊 查询 和 查询 关键 字 描 红 功能 。 

后 台 登 录入 口 ， 为 管理 员 进 入 后 台 提 供 一 个 入 口 。 

下 面 看 一 下 本 案例 中 提供 的 前 台 首 页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\01\99pursey\index.php， 

如 图 1.59 所 示 。 


Hi: E 


MAR , EAR SRE, EE 
和 本 肛交 ,家 好 三， 让 声 ， 可 月 交房 间 # 内 入 均 
ee 和 


ABCDEFG:... 


明日 夜校 教 你 说 最 正宗 的 英语 日语 


大 刀 放 局 于 们 息 同 
ES 
于 系 昌 活 : sr09 
Ree en, Wat 正本 贸 ， 纪 本 机。 由 视 极 ，24 小 时 交 热 水 
“安全 卫生 。 交 钴 多 各 界 识 订 质 弟 自 男士 


澶 要 兴 泊 ,2 楼 。 尖 供 2 人 问 ,9 人间。 交 浊 傍 利 ， 室 内 干净 可 洁 ， 渤 风 妈 ,有 腿 生 位， 
和 电流， 将近 本 - 人 生理， 全 生化 二 人 生理 是- 


图 1.59 九 九 度 供求 信息 网 首页 
各 区 域 的 介绍 及 所 对 应 的 PHP 文件 如 表 1.3 所 示 。 


表 1.3 页 面 框架 中 各 区 域 介绍 及 对 应 的 PHP 文件 


名 称 
导航 栏 

信息 检索 区 
内 容 显示 区 


说 明 
提供 查看 各 类 信息 的 超 链接 
企业 广告 显示 及 各 类 信息 检索 
根据 用 户 请 求 显 示 相 应 内 容 
显示 版 权 信息 


对 应 的 PHP 文件 


根据 请 求 加 载 相应 的 PHP 文件 ， 默 认 加 载 main.php 
bottom 
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1.7.2 前台 首 页 技术 分 析 


超 链接 在 本 质 上 属于 一 个 网 页 的 一 部 分 ， 它 是 一 种 允许 用 户 同 其 他 网 页 或 站 点 之 间 进 行 连接 的 元 
素 。 各 个 网 页 链接 在 一 起 后 ， 才 能 真正 构成 一 个 网 站 。 按 照 使 用 对 象 的 不 同 ， 网 页 中 的 链接 可 以 分 为 
文本 超 链 接 、 图 像 超 链接 、E-mail 链接 、 锚 点 链接 、 多 媒体 文件 链接 和 空 链接 等 。 

图 像 不 但 可 以 建立 超 链 接 ， 还 可 以 实现 图 像 映射 。 图 像 映射 是 指 一 幅 图 像 可 以 建立 多 个 超 链接 ， 
即 在 图 像 上 定义 多 个 区 域 , 每 个 区 域 链接 到 不 同 的 地 址 , 这 样 的 区 域 称 为 热 区 (也 称 为 热点 ，Hotspot)。 
本 项 目 就 是 应 用 图 像 热 区 实现 的 功能 导航 。 

图 像 映 射 有 两 种 : 服务 器 端 映射 (Server-side Image Map) 和 客户 端 映射 (Client-side Image Map )。 
目前 使 用 最 多 的 是 客户 端 映 射 , 因为 客户 端 映 射 使 图 像 上 对 应 区 域 的 坐标 以 及 链接 的 URL 地 址 都 在 浏 
览 器 端 读 入 ， 省 去 和 服务 器 之 间 互 传 坐标 和 URL 的 时 间 。 

在 PHP 文件 中 ,使 用 <MAP> 标 记 创建 图 像 映射 。 语 法 如 下 : 

<IMG SRC= “file name” USEMAP= “#MapName” > 

<MAP NAME= “MapName” > 


<AREA SHAPE=“value” COORDS= “坐标 ” HREF=“URL” ALT=“ 描 述 文字 ”> 
<AREA SHAPE=“value” COORDS=“ 坐 标 ” HREF=“URL” ALT=“ 描 述 文字 ”> 


</MAP> 


在 <IMG> 标 记 中 设置 属性 USEMAP， 确 定 创建 图 像 映射 。 
<MAP> 标 记 的 属性 如 表 1.4 所 示 。 


表 1.4 <MAP> 标 记 的 属性 


<MAP> 标 记 的 属性 描述 
NAME 图 像 映射 的 名 称 
SHAPE 定义 图 像 映射 区 域 的 名 称 
COORDS, 设 定 区 域 坐标 
HREF 设 定 区 域 的 链接 地 址 
AIT 设 定 区 域 链接 的 描述 文字 


在 <MAP> 标 记 中 ， 根 据 属性 SHAPE 的 取 值 不 同 ， 相 应 坐标 的 设 定 也 不 同 。 下 面 介绍 属性 SHAPE 
的 3 种 取 值 以 及 相应 坐标 的 设 定 。 
设 定 属性 SHAPE 的 属性 值 为 rect。 属 性 SHAPE 取 值 为 rect， 表 示 和 矩形 区 域 ， 属 性 COORDS 
的 坐标 形式 为 “X1，Y1，X2，Y2”。 其 中 ，X1、Y1 代表 矩形 左上 角 的 X 坐标 和 YY 坐标 ，X2、 
Y2 代表 矩形 右 下 角 的 X 坐标 和 YY 坐标。 
设 定 属性 SHAPE 的 属性 值 为 circle。 属性 SHAPE 取 值 为 circle, 表示 圆 形 区 域 , 属性 COORDS 
的 坐标 形式 为 “X，Y，r”。 其 中 ，X、Y 为 圆心 坐标 ，r 为 圆 的 半径 。 
设 定 属性 SHAPE 的 属性 值 为 poly。 属 性 SHAPE 取 值 为 poly, 表示 多 边 形 区 域 ,属性 COORDS 
的 坐标 形式 为 “X1，Y1，X2，Y2，…”。 其 中 ，Xn、Yn 代表 构成 多 边 形 每 一 点 的 坐标 值 ，n 
的 取 值 为 1，2，3，… 多 边 形 有 几 边 就 有 几 对 X、Y 坐标 。 
下 面 在 Dreamweaver 编辑 器 中 创建 图 像 映射 ， 单 击 图 像 不 同 区域 ， 链 接 到 不 同 的 文件 。 创 建 图 像 


@ 


加 
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映射 的 步骤 如 下 : 

(1) 在 Dreamweaver 编辑 器 中 选中 所 要 编辑 的 图 像 ， 在 属性 窗口 中 ， 有 3 种 形状 的 热点 工具 ,分 
别 为 矩形 热点 工具 、 圆 形 热点 工具 、 多 边 形 热点 工具 。 

(2) 选择 一 种 热点 工具 ， 在 图 像 上 按 住 鼠标 左 键 拖 动 ， 确 定 所 选区 域 。 

(3) 设置 区 域 属 性 ， 在 “链接 ”文本 框 中 输入 所 要 链接 书签 的 名 称 ， 在 “ 蔡 代 ”文本 框 中 输入 区 
域 链接 文件 ， 如 index.php。 

(4) 重复 步骤 (2) 和 步骤 (3)， 在 图 像 上 定义 不 同 的 区 域 链接 。 


1.7.3 ”前台 首页 的 实现 过 程 


本 系统 中 所 有 的 前 台 页 面 都 采用 了 二 分 栏 结构 ， 分 为 导航 栏 、 信 息 检索 区 、 内 容 显示 区 和 版 权 区 4 
个 区 域 。 为 了 方便 网 站 的 日 后 维护 ， 将 这 4 个 区 域 形成 单独 的 PHP 文件 ， 然 后 应 用 include 语句 将 这 4 
个 文件 包含 进来 。 前 台 首 页 文件 的 代码 如 下 : 

倒 性 01 ”代码 位 置 ， 光盘 \TM\01\99pursey\index.php 


<table width="780" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr valign="top"> 
<td colspan="2"><?php include("top.php");?></td> < 上 -包含 导航 文件 -> 
</tr> 
<tr> 
<!-- 包 含 信息 检索 文件 --> 
<td width="217" valign="top" background="Images/line2.gif’><?php include("left.php");?></td> 
<!-- 包 含 内 容 显 示 文 件 --> 
<td width="586" valign="top" bgcolor="#FEFEF6"><?php include("main.php");?></td> 
</tr> 
<tr> 
<td colspan="2"><?php include("bottom.php");?></td> < 上 -包含 版 权 信息 文件 -> 
</tr> 
</table> 


其 中 ， 导 航 文件 top.php 页 应 用 了 图 像 映射 来 创建 文件 的 超 链接 ， 代 码 如 下 : 
倒 程 02 代码 位 置 ， 光盘 \TM\01\99pursey\top.php 
<table width="780" height="201" border="0" align="center" cellpadding="0" cellspacing="0"> 


<tr> 
<td height="38"><img src="Images/banner.gif" width="780" height="202" border="0" usemap="#Map"></td> 
</tr> 
</table> 
I 图 像 映射 一 一 一 一 一 一 一 一 -> 
<map name="Map"> 
二 一 -一 联系 我 们 -一 -一 一 一- 一 > 
<area shape="rect" coords="685,8,744,27" href="mailto:99pursey@pursey**.com"> 
ed 投 为 首页 二 一 -> 


<area shape="rect" coords="613,9,666,27" 
‘onClick="this.style.behavior='url(#default#homepage)';this.setHomePage('http://localhost/99pursey/index.php');"> 
一 一 一 一 加 入 收藏 -一 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 -- > 
<area shape="rect" coords="535,8,593,26" 
href="javascript:window.external.AddFavorite('http://localhost/99pursey/index.php',' 九 九 度 供求 信息 网 ')"> 


@ 


< 1- -一 一 一 一 一 一 一 一 一 一 一 - 


营 ] 音 
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<area shape="rect" coords="448,8,516,26" href="admin/login.php"> < 上 -后 台 登 录 页 --> 
<area shape="rect" coords="356,8,424,28" href="release.php"> <!- 发 布 信息 页 --> 
<area shape="rect" coords="680,61,753,82" href="search.php"> <!-- 寻 人 / 物 启 示 信 息 页 -> 
<area shape="rect" coords="579,62,637,82" href="sale.php"> <!- 出 售 信息 页 --> 
<area shape="rect" coords="483,62,536,83" href="car.php"> < 上 -车 辆 信息 页 --> 
<area shape="rect" coords="385,62,441,83" href= "teaching.php"> <!- 家 教 信息 页 --> 
<area shape="rect" coords="290,61,344,83" href="seekjob.php"> < 上 -求职 信息 页 --> 
<area shape="rect" coords="678,38,740,58" href="recruitbusiness.php"> <!- 招 商 引资 信息 页 --> 
<area shape="rect" coords="579,38,635,58" href="seekbuy.php"> < 上 -求购 信息 页 --> 
<area shape="rect" coords="482,38,541,58" href="house.php"> <!- 房 屋 信息 页 --> 
<area shape="rect" coords="381,39,440,60" href="foster.php"> <!- 培 训 信息 页 --> 
<area shape="rect" coords="290,39,345,59" href="invitejob.php"> <!- 招 聘 信息 页 --> 
<area shape="rect" coords="204,39,263,86" href="index.php"> <!- 首 页 -> 

</map> 


1.8 免费 供求 信息 发 布 模块 设计 


1.8.1 免费 供求 信息 发 布 模块 概述 


免费 供求 信息 的 发 布 提 供 对 象 为 供求 信息 用 户 ， 是 供求 信息 网 站 非常 重要 的 功能 ， 也 是 供求 信息 
网 站 的 核心 功能 。 

免费 供求 信息 发 布 模块 可 以 完成 11 种 不 同类 别 信息 的 发 布 。 用 户 可 以 根据 自身 需要 将 供求 信息 发 
布 到 相应 的 信息 类 别 中 〈 共 包括 11 个 信息 类 别 : 公寓 信息 、 招 聘 信 息 、 求 职 信息 、 培 训 信息 、 家 教 信 
息 、 房 屋 信息 、 车 辆 信息 、 求 购 信 息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启示 等 类 型 供求 信息 )。 信 息 发 
布 成 功 后 ， 需 要 管理 员 进 行 审核 ， 只 有 审核 成 功 的 信息 才能 显示 在 前 台 相 应 的 信息 类 别 网 页 中 。 免 费 
供求 信息 发 布 的 流程 如 图 1.60 所 示 。 


在 前 台 展 示 
发 布 的 信息 


图 1.60 免费 供求 信息 发 布 流程 图 
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1.8.2 ”免费 供求 信息 发 布 模块 技术 分 析 


本 模块 实现 免费 供求 信息 发 布 功 能 ， 主 要 应 用 到 如 下 几 个 函数 。 
1. mysql_connect() 函 数 


打开 一 个 到 MySQL 服务 器 的 连接 。 如 果 成 功 则 返回 一 个 MySQL 连接 标识 ， 失 败 则 返回 false。 
语法 如 下 : 
resource mysql_connect ( [string server [, string username [, string password [, bool new _link [, int client_flags]]]]] ) 
mysql_connect() 函 数 的 参数 说 明 如 表 1.5 所 示 。 
表 1.5 mysql_connect() 函 数 的 参数 说 明 
参 数 说 了 明 
MySQL 服务 器 。 可 以 包括 端口 号 ， 例 如 "hostname:port"， 或 者 到 本 地 套 接 字 的 路 径 ， 例 如 对 于 localhost 
ee 的 ":/path/to/socket"。 如 果 PHP 指令 mysql.default_host 未 定义 (默认 情况 )， 则 默认 值 是 "localhost:3306' 
username ”| 用 户 名 。 默 认 值 是 服务 器 进程 所 有 者 的 用 户 名 
password 密码 。 默 认 值 是 空 密码 
如 果 用 同样 的 参数 第 二 次 调用 mysql_connect() 函 数 , 将 不 会 建立 新 连接 , 而 是 返回 已 经 打开 的 连接 标识 。 
new_link “| 参数 new_link 改变 此 行为 并 使 mysql_connect0 函 数 总 是 打开 新 的 连接 ， 甚 至 当 mysql_connect() 函 数 曾 
在 前 面 被 同样 的 参数 调用 过 
client_flags 参数 可 以 是 以 下 常量 的 组 合 : MYSQL_CLIENT_ SSL、MYSQL_ CLIENT_COMPRESS、 
MYSQL CLIENT IGNORE _ SPACE 或 MYSQL_ CLIENT INTERACTIVE 


client_flags 


2. mysql_select_db() 函 数 
选择 MySQL 数据 库 。 如 果 成 功 返 回 true， 失 败 返 回 false。 语 法 如 下 : 


bool mysql_select_db ( string database_name [, resource link_identifier] ) 


mysql_select_db() 函 数 设 定 与 指定 的 连接 标识 符 所 关联 的 服务 器 上 的 当前 激活 数据 库 。 如 果 没 有 指 
定 连接 标识 符 , 则 使 用 上 一 个 打开 的 连接 。 如 果 没有 打开 的 连接 , 本 函数 将 无 参数 调用 mysql_connect() 
函数 来 尝试 打开 一 个 并 使 用 之 。 每 个 其 后 的 mysql_query(0) 函 数 调用 都 会 作用 于 活动 数据 库 。 

下 面 应 用 mysql_connect0 函 数 连接 MySQL 服务 器 ， 然 后 应 用 mysql_select_db() 函 数 连 接 MySQL 
数据 库 ， 代 码 如 下 : 


<?php 

$conn = mysql_connect("localhost", "root", "root"); /| 连接 MySQL 服务 器 
$db=mysql_select_db("db_pursey", $conn) or die ("数据 库 连 接 失败 :" .mysql_error()); // 连 接 数 据 库 db _pursey 
mysql_query("set names gb2312"); /采用 GB2312 编码 方式 
9% 


3. mysql_query() 函 数 
mysql_query(O) 函 数 用 来 根据 连接 标识 符 向 该 数据 库 服 务 器 的 当前 数据 库 发 送 查询 。 语 法 如 下 : 


@ 


参 
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int mysql_query(string query ,int [link_identifier]); 


其 中 ，query 是 查询 字符 串 ， link_identifier 是 数据 库 连 接 标识 符 。 
mysql_query() 函 数 在 执行 成 功 时 返回 一 个 结果 标识 符 ， 失 败 时 返回 false。 


4. date() 函 数 
date() 函 数 主要 用 于 格式 化 一 个 本 地 时 间 / 日 期 。 语 法 如 下 : 


string date ( string format , int timestamp) 


该 函数 返回 将 参数 timestamp 按照 指定 格式 格式 化 而 产生 的 字符 串 ,其 中 参数 timestamp 是 可 选 的 ， 
默认 值 为 time0， 即 如 果 没 有 给 出 时 间 戳 ， 则 使 用 本 地 当前 时 间 。 


date0) 函 数 的 参数 format 的 格式 化 选项 如 表 1.6 所 示 。 
表 1.6 参数 format 的 格式 化 选项 
数 说 了 明 
小 写 的 上 午 和 下 午 值 ， 返 回 值 为 am 或 pm 
大 写 的 上 午 和 下 午 值 ， 返 回 值 为 AM 或 PM 
Swatch Internet 标准 时 ， 返 回 值 为 000 一 999 
月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 为 01 一 31 
星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 为 Mon 一 Sun 
月 份 ， 完 整 的 文本 格式 ， 返 回 值 为 January 一 December 
小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 
小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 0 一 23 
12 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 01 一 12 
小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 00 一 23 
有 前 导 零 的 分 钟 数 ， 返 回 值 为 00 一 59 
判断 是 否 为 夏令 时 ， 返 回 值 如 果 是 夏令 时 为 1， 否 则 为 0 
月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 为 1~31 
星期 数 ， 完 整 的 文本 格式 ， 返 回 值 为 Sunday 一 Saturda 
判断 是 否 为 闽 年 ， 返 回 值 如 果 是 闽 年 为 1， 否 则 为 0 
数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 为 01 一 12 


ole ls |= 


“||-:| 工 |=> op 
> 
2 


3 个 字母 缩写 表示 的 月 份 ， 返 回 值 Jan 一 Dec 


数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 


与 格林 威 治 时 间 相 差 的 小 时 数 ， 例 如 0200 
RFC 822 格式 的 日 期 ， 例 如 ，Thu，21 Dec 2000 16: 01: 07+0200 


秒 数 ， 有 前 导 零 ， 返 回 值 为 00 一 59 


每 月 天 数 后 面 的 英文 后 组 ，2 个 字符 ， 例 如 st、nd、rd 或 者 th。 可 以 和 j 一 起 使 用 


指定 月 份 所 应 有 的 天 数 
本 机 所 在 的 时 区 


从 UNIX 纪元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 


星期 中 的 第 几 天 ， 数 字 表示 ， 返 回 值 为 0 一 6 


zz co-|- lvls | lo ls lz |3 Ic I- 


ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 
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续 表 
说 了 明 
4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 
2 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
年 份 中 的 第 几 天 ， 返 回 值 为 0 一 366 
时 差 偏 移 量 的 秒 数 .UTC 西边 的 时 区 偏 移 量 总 是 负 的 ,UTC 东边 的 时 区 偏 移 量 总 是 正 的 ,返回 值 为 -43200~43200 


人 注意 有 效 的 时 间 戳 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54 到 2038 年 1 月 19 
日 03:14:07 (此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 )。 在 Windows 系统 中 此 范围 限制 为 从 
1970 年 1 月 1 日 到 2038 年 1 月 19 日 。 


例如 ， 应 用 date(0) 函 数 格式 化 一 个 日 期 ， 并 输出 日 期 的 值 。 代 码 如 下 : 
<?php 

echo date("y:m:d"); 

?> 


结果 为 : 07:07:11。 
1.8.3 ”免费 供求 信息 发 布 模块 的 实现 过 程 


国 ” ”免费 供 求 信息 发 布 使 用 的 数据 表 : tb_info 

用 户 通过 单 击 前 台 页 面 导航 栏 中 的 “我 要 发 布 ” 超 链接 ， 进 入 信息 发 布 页 面 ， 如 图 1.61 所 示 。 程 
序 会 先 验证 用 户 是 否 输入 信息 ， 车 验证 失败 ， 则 返回 信息 发 布 页 面 ， 进 行 相应 提示 ， 若 验证 成 功 ， 则 
向 数据 库 中 插入 记录 ， 完 成 发 布 操作 。 


图 1.61 免费 供求 信息 发 布 网 页 
在 功能 导航 页 top.php 中 ， 在 “我 要 发 布 ” 图 像 上 添加 图 像 映 射 ， 并 绘制 热 区 。 代 码 如 下 : 
倒 程 03 ”代码 位 置 : 光盘 \TM\01\99pursey\top.php 


<map name="Map"> /| 图像 映射 
<area shape="rect" coords="356,8,424,28" href="release.php"> 


</map> 


在 信息 发 布 页 面 选择 要 发 布 的 信息 类 型 后 ,填写 真实 有 效 的 供求 信息 。 为 了 避免 用 户 添加 空 信息 ， 


@ 
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在 单 击 “ 发 布 信息 ”按钮 时 ， 应 用 JavaScript 脚本 自 定义 一 个 checkform0 函 数 ， 验 证 提交 的 表单 各 元 
素 是 否 为 空 值 ， 如 果 为 室 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。 代 码 如 下 : 
倒 程 04 ”代码 位 置 ， 光盘 \TM\01\99pursey\release_content.php 


<script language="javascript"> 


function checkform(formX{ // 自 定义 一 个 JavaScript 函数 checkform() 
for(i=0;i<form.length;i++X{ /应 用 for 循环 语句 检索 form 表单 元 素 的 值 是 否 为 空 
if(form.elements[i].value==""X{ // 如 果 form 表单 中 某 个 元 素 值 为 空 
alert(" 请 将 发 布 信息 填写 完整 !"); // 弹 出 提示 信息 
form.elements[i] .focus(); // 将 焦点 的 值 定位 到 为 空 值 的 表单 元 素 
return false; /| 返回 焦点 
} 
+ 
和 
</script> 


创建 与 数据 库 db_pursey 的 连接 ， 代 码 如 下 : 
倒 程 05 ”代码 位 置 ， 光盘 \TM\01\99pursey\conn\conn.php 


<?php 
$link=mysql_connect("localhost","root","root"); /连接 MySQL 服务 器 
mysql_select_db("db_pursey ",$link); /连接 MySQL 数据 库 文件 db_pursey 
mysql_query("set names gb2312"); /采用 GB2312 编码 方式 
?> 

Ah) 代码 贴 十 


@ mysql_connect0 函 数 : 连接 MySQL 服务 器 ， 详 细 讲 解 参见 1.8.2 节 。 
@ mysql_select db() 函 数 : 连接 MySQL 数据 库 文 件 ， 详 细 讲解 参见 1.8.2 节 。 
四 mysql_query() 函 数 : 用 来 向 数据 库 服务 器 发 送 编码 方式 ， 详 细 讲解 参见 1.8.2 节 。 


提交 表单 信息 到 数据 处 理 页 ， 应 用 insert…into 语句 向 免费 供求 信息 表 中 添加 供求 信息 。 如 果 信 息 
添加 成 功 ， 则 弹出 成 功 的 信息 提示 ; 否则 弹出 失败 的 提示 信息 。 代 码 如 下 : 
倒 程 06 ”代码 位 置 : 光盘 \TM\01\99pursey\release_ok.php 


<?php 

include("conn/conn.php"); /连接 数据 库 文件 
qtype=$_POST[type]; /获取 信息 类 型 
qtitle=$_POSTItitle], /获取 信息 标题 
$content=$_POST[content]; // 获 取信 息 内 容 
S$linkman=$_POST[linkman]; /获取 联系 人 
S$tel=$_POSTItel]; /获取 联系 电话 
© $edate=date("Y-m-d hii:s"); // 获 取 发 布 时 间 


@ $sq=mysql_query("insert into tb_info(type,title,content,linkman,tel,checkstate,edate) 

values('$type','$title''$content','$linkman','$tel',0,'$edate')"); // 将 免费 的 供求 信息 添加 到 数据 表 中 

if($sql){ // 如 果 添加 操作 成 功 ， 则 弹出 提示 信息 
echo "<script>alert( 恭 喜 您 ， 信 息 发 布 成 功 ! ');window.location.href='release.php';</script>"; 

}else{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert( 对 不 起 ， 信 息 发 布 失败 ! ');history.back();</script>"; 


% 
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< 代码 贴 十 
@ date0 函 数 : 格式 化 一 个 本 地 时 间 / 日 期 ， 详 细 讲 解 参 见 1.8.2 节 。 
@ insert into: 向 指定 的 数据 表 中 添加 数据 信息 。 


1.9 ”信息 检索 模块 设计 


1.9.1 信息 检索 模块 概述 


信息 检索 是 对 已 存在 于 数据 库 中 的 数据 按 条 件 进 行 筛选 浏览 ， 是 查看 历史 信息 和 确认 数据 操作 最 
为 快速 、 有 效 的 办 法 。 信 息 检索 模块 主要 通过 选择 信息 类 型 和 输入 查询 关键 字模 糊 查询 供求 信息 资源 ， 
并 输出 查询 结果 。 考 虑 到 供求 信息 的 信息 量 较 大 ， 因 此 本 模块 对 与 查询 关键 字 相 匹配 的 查询 结果 进行 
描 红 ， 从 而 方便 用 户 的 浏览 。 信 息 检 索 模 块 的 示意 图 如 图 1.62 所 示 。 


设置 查询 条 件 


输出 查询 结果 | 是 | 1 a 
Ti 根据 检索 式 查找 数据 Ln 输出 提示 信息 


图 1.62 信息 检索 模块 的 示意 图 
1.9.2 ”信息 检索 模块 技术 分 析 


在 对 数据 进行 查询 后 ， 最 终 需 要 将 查询 结果 显示 在 页 面 中 反馈 给 浏览 者 。 在 PHP 中 ， 查 询 结果 的 
显示 方式 有 很 多 种 ， 最 常用 的 就 是 表格 显示 方式 。 因 为 采用 这 种 方式 显示 的 数据 条 理 清晰 、 简 洁 明 了 。 

在 利用 表格 显示 查询 结果 时 , 通常 是 将 查询 结果 保存 在 结果 集中 , 然后 需要 使 用 do…while 循环 将 
其 查询 结果 显示 出 来 。 需 要 注意 的 是 ， 需 要 先 判断 查询 结果 是 否 为 室 ， 只 有 查询 结果 不 为 空 时 ， 才 可 
以 使 用 循环 语句 显示 数据 。 为 了 使 读者 更 好 地 理解 通过 表格 显示 查询 结果 ， 下 面 给 出 其 实现 流程 图 ， 
如 图 1.63 所 示 。 


@ 


1.63 ”信息 检索 模块 流程 图 


考虑 到 用 户 不 可 能 全 面 了 解数 据 表 中 的 数据 信息 ， 例 如 不 能 确定 所 要 查询 信息 的 内 容 、 查 询 的 主 
题 等 ,这 时 就 需要 使 用 like 进行 模糊 查询 。like 关键 字 需 要 使 用 通配符 在 字符 串 内 查找 指定 的 模式 ， 所 
以 读者 需要 了 解 通配符 及 其 含义 。 通 配 符 的 含义 如 表 1.7 所 示 。 

表 1.7 like 关键 字 中 的 通配符 及 说 明 


由 零 个 或 更 多 字符 组 成 的 任意 字符 串 


任意 单个 字符 
用 于 指定 范围 ， 例 如 [A~~F] 表 示 A~F 范围 内 的 任何 单个 字符 
表示 指定 范围 之 外 的 ， 例 如 [^A 一 F] 表 示 A~F 范围 以 外 的 任何 单个 字符 


如 果 想 查询 包含 “女子 公寓 ”的 信息 ， 可 以 使 用 like 运算 符 配 合 通配符 “%” 完 成 。 其 SQL 语句 
如 下 : 


select * from tb_info where content like '% 女 子 公寓 %'; 


如 果 想 查找 信息 类 型 为 “公寓 信息 ”或 者 内 容 为 “女子 公寓 ”的 信息 时 配合 or 运算 符 来 使 用 。 其 
SQL 语句 如 下 : 


select * from tb_info ”where type=' 公 寓 信 息 ' or content like'% 女 子 公寓 %' 
本 模块 实现 付费 信息 与 查询 关键 字 相 匹配 的 信息 的 SQL 语句 如 下 : 


$type=$_POST[type]; /获取 信息 类 型 
$content=$_POST[content]; // 获 取 查 询 关 键 字 
$sql1=mysql_query("select * from tb leaguerinfo where checkstate=1 and type='$type' and content 
like'%$content%' or title like'% $content%' or linkman like'%$content%' or tel like'% $content%"); 
$info1=mysql_fetch_array($sql1); /检索 付费 的 信息 


本 模块 实现 免费 信息 与 查询 关键 字 相 匹 配 的 信息 的 SQL 语句 如 下 : 
$sql=mysql_query("select * from tb_info where checkstate=1 and type='$type' and content like'% $content%' or 


title like'% $content%' or linkman like'%$content%' or tel like'%$content%"™"); 
$info=mysql_fetch_array($sql); /检索 免费 的 信息 
@ 
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OE 


另外 ， 由 于 搜索 的 内 容 中 文字 比较 多 ， 为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 信息 ， 所 以 在 搜索 
引擎 中 加 入 了 描 红 功能 。 描 红 功 能 主要 用 str_ireplace() 函 数 实现 , 该 函数 的 具体 讲解 读者 可 参见 本 章 的 
1.14.1 节 。 


1.9.3 ”信息 检索 模块 的 实现 过 程 


国 信息 检索 模块 使 用 的 数据 表 : tb_info、tb_leaguerinfo 

在 开发 信息 检索 模块 时 ， 由 于 该 网 站 含有 大 量 的 数据 
信息 ， 为 了 方便 用 户 浏 览 网 站 信息 ， 需 要 添加 复合 条 件 查 
询 实现 搜索 功能 。 在 信息 检索 区 的 “关键 字 ” 文 本 框 中 输 
入 欲 查 询 的 关键 字 ， 在 “条 件 ” 下 拉 列 表 框 中 选择 要 搜索 
的 信息 类 型 ， 然 后 单 击 “ 开 始 搜索 ”按钮 ， 对 指定 条 件 的 
记录 进行 检索 并 输出 结果 集 到 浏览 器 ， 同 时 为 了 方便 浏览 
者 查找 自己 所 关注 的 内 容 信息 ， 本 模块 对 查询 关键 字 进 行 
描 红 。 运 行 结果 如 图 1.64 所 示 。 1.64 ”信息 检索 页 面 的 运行 结果 

信息 检索 页 面 中 所 涉及 的 重要 表单 元 素 如 表 1.8 所 示 。 


表 1.8 信息 检索 页 面 所 涉及 的 重要 表单 元 素 


<select name= "type"> 
<option value=" 招 聘 信 息 ">- 招 聘 信 息 -</option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 -</option> 


Src="Images/btnl.gif" onClick="retum chkinput 开始 搜索 按钮 


应 用 JavaScript 脚本 自 定 义 一 个 chkinput() 函 数 ， 实 现 对 表单 提交 的 信息 进行 验证 。 代 码 如 下 : 
倒 程 07 ”代码 位 置 ， 光盘 \TM\01\99pursey\left.php 


<script language="javascript"> 


function chkinput(formX{ // 自 定义 一 个 chkinput() 函 数 
if(form.content.value=="" // 判 断 如 果 查 询 关键 字 文本 框 等 于 空 
alert(" 请 输入 查询 关键 字 !"); // 则 弹出 提示 信息 
form.content.select(); /重新 定位 焦点 
return false; // 返 回 表单 元 素 
} 
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将 表单 信息 提交 到 数据 处 理 页， 连接 数据 库 文件 ， 接 收 表 单 信息 ， 然 后 用 mysql_query0 函 数 向 服 
务 器 发 送 SQL 语句 ， 检 索 与 查询 关键 字 相 匹配 的 信息 资源 。 代 码 如 下 : 
倒 各 08 ”代码 位 置 ， 光盘 \TM\01\99pursey\findinfo.php 


<?php 

include("conn/conn.php"); // 连 接 数 据 库 文件 
$type=$_POSTI[type]; /获取 信息 类 型 
$content=$_POST[content]; // 获 取 查 询 关键 字 


$sql1=mysql_query("select * from tb_leaguerinfo where checkstate=1 and type='$type' and content like'%$content%' 
or title like'% $content%' or linkman like'%$content%' or tel like'%$content%"); 
$info1=mysql_fetch_array($sql1); /检索 付 费 的 供求 信息 
$sql=mysql_query("select * from tb_info where checkstate=1 and type='$type' and content like'%$content%' or 
title like'%$content%' or linkman like'%$content%' or tel like'% $content%"™"); 

$info=mysql_fetch_array($sql); /检索 免费 的 供求 信息 


?> 


V4 
说明 信息 检索 需要 从 免费 供求 信息 表 tb_info 和 付费 供求 信息 表 tb_leaguerinfo 中 获取 数据 , 因 
此 需要 向 MySQL 服务 器 传递 两 条 SQL 语句 。 


用 do…while 循环 语句 输出 付费 信息 与 查询 关键 字 相 匹配 的 信息 资源 ， 并 用 str_ireplace0) 函 数 对 查 
询 关 键 字 实现 描 红 功 能 。 代 码 如 下 : 

倒 程 09 ”代码 位 置 ， 光盘 \TM\01\99pursey\findinfo.php 

<!-- 下 面 输出 的 是 付费 信息 与 查询 关键 宇 相 匹配 的 信息 --> 


<?php 
if($info1){ // 如 果 检 索 到 了 付费 信息 
dof // 则 用 do…while 循环 语句 输出 付费 信息 
?> 
<table width="540" border="0" cellspacing="0" cellpadding="0"> 
<tr> 


<td height="26"> 

<!-- 应 用 str_ireplace() 函 数 对 查询 关键 字 进 行 描 红 一 > 

<! 一 对 与 查询 关键 字 所 匹配 的 信息 类 型 进行 描 红 --> 

[<?php echo str_ireplace( $content,"<font color=#FF0000'>".$content."</font>", $info1[type]);?>] nbsp; 

<! 一 对 与 查询 关键 字 所 匹配 的 信息 标题 进行 描 红 --> 

<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[tite]);?>&nbsp;&nbsp; 

<! 一 对 与 查询 关键 字 所 匹配 的 发 布 时 间 进 行 描 红 --> 

<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[edate]);?></td> 
</tr> 
<tr> 

<td height="26">&nbsp;&nbsp;&nbsp;&nbsp; 

<! 一 对 与 查询 关键 字 所 匹配 的 信息 内 容 进行 描 红 --> 

<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[content]);?></td> 
</tr> 
<tr> 

<td height="26">&nbsp; 联 系 人 : 

<!-- 对 与 查询 关键 字 所 匹配 的 联系 人 进行 描 红 -> 

<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>", $info1llinkman]); ?>&nbsp; 


@ 
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<! 一 对 与 查询 关键 字 所 匹配 的 联系 电话 进行 描 红 --> 
联系 电话 : <?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[tel]);?> 
<! 一 - 一 > 
</td> 
</tr> 
</table> 
<?php 
jwhile($info1=mysql_fetch_array($sql1)); /循环 语句 结束 


i 


/ 
后 曙 免费 信息 的 输出 方式 与 付费 信息 的 基本 类 似 ， 代 码 部 分 略 ， 详 见 本 书 附 赠 光 盘 ， 


免费 信息 的 输出 方式 与 付费 信息 的 基本 类 似 ， 下 面 给 出 实现 过 程 的 核心 代码 结构 。 
倒 性 10 代码 位 置 ， 光盘 \TM\01\99pursey\findinfo.php 


a 下 面 输出 的 是 免费 信息 与 查询 关键 字 相 匹配 的 信息 -一 一 -一 -> 
<?php 
if($info)f // 如 果 检 索 到 了 免费 信息 

dof 1/ 则 用 do…while 循环 语句 输出 付费 信息 
J 

// 免 费 信息 的 输出 方式 与 付费 信息 的 类 似 , 代码 略 
<?php 

} while($info=mysql_fetch_array($sql)); /ldo…while 循环 语句 结束 

/ 放 条 件 语句 结束 

?> 


如 果 在 免费 供求 信息 表 和 付费 供求 信息 表 中 没有 检索 到 与 查询 关键 字 相 匹配 的 数据 ， 则 弹出 提示 
信息 。 代 码 如 下 : 
倒 程 11 ”代码 位 置 光盘 \TM\01\99pursey\findinfo.php 


<table width="540" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td align="center"> 您 检索 的 信息 资源 不 存在 ! </td> 
</tr> 
</table> 


1.10.1 后 台 首 页 概述 


程序 开发 人 员 在 设计 网 站 后 台 首页 时 ， 主 要 从 后 台 管 理 人 员 对 功能 的 易 操 作 性 、 实 用 性 、 网 站 的 
易 维 护 性 考虑 ， 因 此 采用 了 框架 技术 。 九 九 度 供求 信息 网 后 人 台 首 页 主要 包含 以 下 内 容 : 
发 布 付费 的 供求 信息 〈 包 括 公寓 信息 、 招 聘 信 息 、 求 职 信息 、 培 训 信息 、 家 教 信息 、 房 屋 信 
息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启示 等 )， 以 及 付费 信息 的 浏览 、 
审核 及 删除 功能 。 


@ 
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免费 信息 的 浏览 、 审 核 及 删除 功能 。 
企业 广告 信息 的 发 布 、 浏 览 、 前 台 推荐 显示 、 删 除 功能 。 
“网 站 首页 ” 超 链 接 ， 为 管理 员 进 入 前 台 提 供 一 个 入 口 。 
“退出 登录 ” 超 链 接 ， 用 于 注销 当前 用 户 。 
本 案例 中 提供 的 后 台 首 页 如 图 1.65 所 示 。 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\01\99pursey\admin\ 
index.php。 
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图 1.65 供求 信息 网 站 后 台 首页 


1.10.2 后台 首 页 技术 分 析 


九 九 度 供求 信息 网 后 台 采 用 框架 技术 进行 页 面 布局 。 所 谓 框架 就 是 网 页 的 各 部 分 为 相互 独立 的 网 
页 ， 又 由 一 个 网 页 将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 者 的 浏览 器 中 ， 重 复出 现 的 内 
容 被 固定 下 来 ， 每 次 浏览 者 发 出 对 页 面 的 请 求 时 ， 只 下 载 发 
生变 化 的 框架 页 面 ， 其 他 子 页 面 保持 不 变 。 SS 

使 用 框架 可 以 将 容器 窗口 划分 为 若干 个 子 窗口 ， 在 每 个 人 
子 窗口 可 以 分 别 显示 不 同 的 网 页 。 首 先 在 Dreamweaver 8 中 © 
创建 一 个 “ 左 一 中 一 右 ”的 框架 集 , 然后 在 标识 中 添加 “上 
一 中 一 下 ”的 框架 集 ， 最 后 在 标识 @ 中 添加 一 个 “ 左 一 右 ” 1 ~ 
的 框架 集 ， 从 而 完成 一 个 完整 的 后 台 框 架 。 构 建 框架 的 流程 
如 图 1.66 所 示 。 

使 用 框架 可 以 非常 方便 地 完成 导航 工作 。 下 面 详细 介绍 
框架 网 页 的 基本 结构 、 设 置 框架 集 的 属性 和 设置 框架 的 属性 。 


1. 框架 网 页 的 基本 结构 


框架 网 页 通过 一 个 或 多 个 ffameset 和 frame 标记 来 定义 。 在 框架 网 页 中 , 将 frameset 标记 置 于 head 
标记 之 后 ， 以 取代 body 标记 的 位 置 ， 当 客户 端 浏 览 器 不 支持 框架 网 页 时 , 还 可 以 使 用 noframes 标记 给 
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1.66 ”网 站 后 台 框 架 流程 
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出 框架 不 能 被 显示 时 的 替换 内 容 。 框 架 网 页 的 基本 结构 如 下 : 


<html> 
<head> 
<title> 基 本 框架 页 </title> 
</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 
<body> 
对 不 起 ! 您 的 浏览 器 不 支持 框架 页 面 的 显示 ! 
</body> 
</noframes> 
</html> 


2. 设置 框架 集 的 属性 


框架 集 包 含 如 何 组 织 各 个 框架 的 信息 ， 可 以 通过 frameset 标记 来 定义 。 框 架 是 按照 行 和 列 来 组 织 
可 以 使 用 frameset 标记 的 下 列 属性 对 框架 的 结构 进行 设置 。 

(1) 左右 分 割 窗口 属性 cols 

在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 左右 分 割 窗口 属性 cols 实现 。 其 语法 格 
式 如 下 : 


<frameset cols="value,value,..."> 
<frame> 
<frame> 

</frameset> 


其 中 ，value 用 于 指定 各 个 框架 的 列 宽 ， 取 值 有 像素 、 百 分 比 〈%) 和 相对 尺寸 (*) 3 种 形式 。 
例如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 列 ， 其 中 第 1 列 占 浏览 器 窗口 宽度 的 20%， 第 2 列 为 
120 像素 ， 第 3 列 为 浏览 器 窗口 剩余 部 分 的 框架 。 代 码 如 下 : 


全 


<frameset cols="20%,120,” > 
<frame> 
<frame> 

</frameset> 


区 5 如 果 将 cols 属性 设置 为 “*，*，*”， 则 表示 将 窗口 划分 成 3 个 等 宽 的 框架 ， 如 果 将 cols 
属性 设置 为 “*，2*，3*”， 则 表示 左边 的 框架 占 窗口 宽度 的 1/6， 中 间 的 框架 占 窗口 宽度 的 1/3， 右 
边 的 框架 占 窗口 宽度 的 1/2。 


(2) 上 下 分 割 窗口 属性 rows 
在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 上 下 分 割 窗口 属性 rows 实现 。 其 语法 格 
式 如 下 : 
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<frameset rows="value,value,..."> 
<frame> 
<frame> 

</frameset> 


其 中 value 用 于 指定 各 个 框架 的 行 高 ， 取 值 有 像素 、 百 分 比 〈%) 和 相对 尺寸 (*) 3 种 形式 , 设置 
方法 与 COLS 属性 类 似 。 例 如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 行 ， 其 中 的 第 1 行 占 浏 览 器 窗口 
宽度 的 20%， 第 2 行为 120 像素 ， 第 3 行为 浏览 器 窗口 剩余 部 分 的 框架 。 代 码 如 下 : 


<frameset rows="20%,120,” > 
<frame> 
<frame> 

</frameset> 


(3) 框架 边框 显示 属性 frameborder 

该 属性 用 于 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框， 默认 值 ) 或 0 (不 显示 边框 )。 

(4) framespacing 

该 属性 用 于 指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 如 果 不 设 置 该 属性 ， 则 框架 之 间 没 有 间隔 。 

(5) 指定 边框 宽度 属性 border 

该 属性 用 于 指定 边框 的 宽度 ， 只 有 frameborder 属性 为 ! 时 有 效 。 

3. 设置 框架 的 属性 

使 用 FRAME 标记 可 以 设置 框架 的 属性 ， 包 括 框 架 的 名 称 、 框 架 是 否 包 含 滚动 条 以 及 在 框架 中 显 
示 的 网 页 等 。 其 语法 格式 如 下 : 


<frame name=" 框 架 名 称 " src=" 文 件 " frameborder=" 数 值 " scrolling=" 值 " [noresize] > 


属性 说 明 如 下 。 

name: 指定 框架 的 名 称 。 

src: 指定 在 框架 中 显示 的 网 页 文件 (包括 HTML、ASP 等 网 页 文件 )。 

frameborder: 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 ) 或 0 (不 显示 )。 默 认 值 为 1。 
scrolling: 指定 框架 是 否 包含 滚动 条 。 如 果 将 该 属性 设置 为 yes， 则 框架 包含 滚动 条 ; 若 将 该 
属性 设置 为 no， 则 框架 不 包含 滚动 条 ;如 果 将 该 属性 设置 为 aato， 则 在 需要 时 包含 滚动 条 。 
noresize: 可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 。 


1.10.3 后台 首 页 的 实现 过 程 


根据 1.10.1 节 和 1.10.2 节 的 页 面 概述 及 实现 技术 分 析 ， 需 要 分 别 创建 实现 各 区 域 的 PHP 文件 ， 如 
实现 Banner 广告 栏 的 top.php、 功 能 导航 栏 的 leftphp、 内 容 显示 区 的 main.php 和 页 尾 文件 bottom.php 
等 。 实 现 该 系统 后 台 框 架 布局 的 完整 代码 如 下 : 

全 性 12 ”代码 位 置 : 光盘 \TM\01\99pursey\admin\index.php 


<frameset rows="*" cols="1*,1005,1” framespacing="0" frameborder="NO" border="0"> 
<frame src="blank.php" name="left" scrolling="NO" noresize> <!-- 设 置 空 框 架 页 --> 


办 办 轨 


PHP 项 目 开发 全 程 实录 (第 3 版 ) 


<frameset rows="1005,*" cols="*" framespacing="0" frameborder="NO" border="0"> 
<frameset rows="94,*,190" cols="*" framespacing="0" frameborder="NO" border="0"> 
<frame src="top.php" name="topFrame" scrolling="NO" noresize> 
<frameset rows="*" cols="229,*"” framespacing="0" frameborder="NO" border="0"> 
<frame src="left.php" name="leftFrame" scrolling="NO" noresize> 
<frame src="main.php" name="mainFrame" scrolling="NO" noresize> 
</frameset> 
<frame src="bottom.php" name="bottomFrame" scrolling="NO" noresize> 
</frameset> 
<frame src="blank.php" name="right" scrolling="NO" noresize> 
</frameset> 
<frame src="blank.php"></frameset> <!-- 设 置 空 框 架 页 -> 
<noframes><body> 
</body></noframes> 


和 8 注意 在 建设 Web 网 站 时 ， 如 何 让 不 同 分 辩 率 的 用 户 痢 能 看 到 网 页 的 最 住 效果 是 程序 员 在 设计 
之 初 所 要 考虑 的 首要 问题 .为 了 使 屏幕 的 分 辨 率 在 大 于 1024 x 768 像素 的 设置 时 仍然 处 于 居中 显示 ， 
只 需要 在 设置 框架 布局 时 ， 在 主 框架 两 侧 各 设置 一 个 宽度 相同 的 blank php 空 页 即 可 。 


1.11 付费 供求 信息 发 布 模块 设计 


1.11.1 付费 供求 信息 发 布 模块 概述 


付费 供求 信息 的 发 布 提 供 对 象 为 供求 信息 用 户 ， 是 供求 信息 网 
站 非常 重要 的 功能 ， 也 是 供求 信息 网 站 的 盈利 点 。 企 业 或 用 户 可 以 从 
根据 自身 需要 对 供求 信息 先进 行 付费 ， 付 费 后 由 管理 员 在 后 台 将 供 管理 员 
求 信息 发 布 到 相应 的 信息 类 别 中 ( 共 包括 11 个 信息 类 别 招聘 信息 、 
求职 信息 、 培 训 信息 、 公 寓 信 息 、 家 教 信息 、 车 辆 信息 、 物 品 求购 、 发 布 的 信息 
物品 出 售 、 求 兑 出 竞 、 企业 广告 等 类 型 供求 信息 )。 供 求 
信息 成 功 发 布 后 ， 管 理 员 需 要 在 后 台 对 发 布 的 供求 信息 进行 审核 , 管理 
如 果 审 核 通过 后 ， 则 显示 在 前 台 相 应 的 信息 类 别 网 页 中 。 付 费 供求 更 
信息 发 布 的 流程 如 图 1.67 所 示 。 
一 
1.11.2 ”付费 供求 信息 发 布 模块 技术 分 析 
在 前 台 展 示 
发 布 的 信息 


付费 供求 信息 与 免费 供求 信息 不 同 的 是 ， 付 费 供求 信息 不 仅 需 
要 收取 一 定 的 费用 ， 而 且 还 需要 一 定 的 时 间 限 制 ， 例 如 ， 网 站 要 求 。 图 1.67 付费 供求 信息 发 布 流程 图 
一 个 月 〈 按 30 天 计算 ) 每 条 信息 交 10 元 的 信息 费 ， 如 果 用 户 交纳 
20 元 ， 那 么 信息 显示 的 天 数 就 是 60 天 。 在 前 台 进行 显示 时 ， 不 需要 管理 员 进行 手动 管理 ， 而 是 通过 程 
序 直 接 计 算出 信息 显示 的 截止 时 间 。 


@ 
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信息 显示 的 截止 时 间 =“ 系 统 当前 日 期 ”+“ 信 息 的 有 效 天 数 〈 与 用 户 交纳 的 信息 费 相关 )”。 
自动 计算 信息 显示 的 截止 时 间 的 具体 代码 如 下 : 


$days=$_POST[days]; // 通 过 表单 传 值 获取 信息 显示 的 天 数 
$showday=date("Y-m-d",(time()+3600*24*$days)); // 信 息 显示 的 截止 时 间 


4 
说 明 信息 的 有 效 天 数 与 用 户 交纳 的 信息 党 相关 ， 交 费 不 通过 本 程序 完成 ， 因 此 ， 信 息 的 有 效 
天 数 需要 管理 员 手 动 添加 。 


1.11.3 “付费 供求 信息 发 布 模块 的 实现 过 程 


国 ”付费 供求 信息 发 布 使 用 的 数据 表 : tb_leaguerinfo 

用 户 通 过 单 击 页 面 导航 区 的 “付费 信息 ” 超 链接 ， 进 入 付费 信息 发 布 页 面 ， 如 图 1.68 所 示 。 填 写 
真实 有 效 的 付费 信息 ， 单 击 “ 发 布 信息 ”按钮 ， 程 序 会 先 验 证 用 户 输入 的 信息 ， 若 验证 失败 ， 则 返回 
信息 发 布 页 面 ， 进 行 相应 提示 ; 车 验 证 成 功 ， 则 向 数据 库 中 插入 记录 ， 完 成 付费 信息 的 发 布 操作 。 
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图 1.68 付费 供求 信息 发 布 页 面 运行 结果 
在 左 侧 框架 left.php 页 中 ， 添 加 “付费 信息 ”图 像 域 及 表单 。 代 码 如 下 : 
倒 程 13 ”代码 位 置 : 光盘 \TM\01\99pursey\admin\left.php 


<form name="form1" method="post" action="release_content.php" target="mainFrame"> 

<input name="imageField" type="image" class="input1" src="images/btn_fufei.gif" width="210" height="39" 
border="0"> 
</form> 


单 击 “ 付 费 信息 ”按钮 ， 将 信息 页 release_content.php 中 的 内 容 显示 在 框架 显示 页 mainFrame 中 。 
付费 供求 信息 发 布 页 面 主要 用 于 发 布 付费 的 供求 信息 , 该 页 面 中 所 涉及 的 重要 表单 元 素 如 表 1.9 所 示 。 


® 
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表 1.9 付费 供求 信息 页 面 所 涉及 的 重要 表单 元 素 


名 称 重要 属性 含义 
forml method="post" action="release_ ok.php” 表单 
<select name= "type"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 -</option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 -</option> 信息 类 型 
<option value=" 寻 人 / 物 启示 ">- 寻 人 / 物 启示 -</option> 
</select> 
flag checkbox class="inputl”_ value="1" checked “是 否 付费 ” 复 选 杠 
title text 信息 标题 
content textarea 信息 内 容 
linkman 联系 人 
tel 联系 电话 
days 有 效 天 数 
imageField “发 布 信息 ”按钮 


在 付费 信息 发 布 页 面 选择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 为 了 避免 用 户 添加 空 
按钮 时 ， 应 用 JavaScript 脚本 自 定义 一 个 
定位 到 为 空 


信息 ， 在 单 击 “ 发 布 信息 ” 
单 各 元 素 是 否 为 空 值 ， 如 果 为 室 ， 则 弹出 提示 信息 ， 并 将 焦 8 
函数 的 代码 部 分 与 例 程 04 相同 ， 这 里 不 再 次 述 。 
提交 表单 信息 到 数据 处 理 页 ， 应 用 insert… 
添加 成 功 ， 则 弹出 成 功 的 信息 提示 ; 否则 弹出 失败 的 提示 信息 。 
倒 程 14 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\release_ok.php 


checkform() 函 数 ， 验 证 提交 的 表 
s 值 的 表单 元 素 。checkform0) 


“into 语句 向 付费 供 求 信 县 表 中 添加 供求 信 息 。 如 果 信 息 


<?php 

include("../conn/conn.php"); // 连 接 数 据 库 文件 
qtype=$_POST[type]; // 获 取信 息 类 型 
$flag=$_POSTIflag]; // 获 取 付 款 状态 
$title=$_POSTItite]; // 获 取信 息 标题 
$content=$_POST[content]; /获取 信息 内 容 
$linkman=$_POSTIlinkman]; // 获 取 联 系 人 
$days=$_POST[days]; // 获 取 发 布 时 间 
S$tel=$_POSTItel]; /| 获取 联系 电话 
$sdate=date("Y-m-d"); // 当 前 系统 时 间 

© $showday=date("Y-m-d",(time()+3600*24*$days)); // 获 取信 息 的 有 效 时 间 


$sql=mysql_query("insert into tb_leaguerinfo(type,title,content,linkman,tel,sdate,showday,checkstate) 
values('$type','$title''$content','$linkman','$tel','$sdate','$showday',$flag)");，// 将 付费 的 供求 信息 添加 到 数据 表 中 
if($sql){ // 如 果 添 加 操作 成 功 ， 则 弹出 提示 信息 
@ ”echo "<script>alert(' 信 息 发 布 成 功 !); parent.mainFrame.location.href='release_content.php';</script>"; 
}else{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert(' 信 息 发 布 失败 ! ");history.back();</script>"; 
局 
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} 


$s 
二 提 代码 贴 十 
@ date("Y-m-d",(time()+3600*24*$days)): 信息 的 有 效 时 间 = 当 前 期 日 期 + 付费 期 限 . 应 用 time() 函 数 获取 当前 日 期 时 


间 蕉 ， 付 费 期 限 的 时 间 蕉 等 于 3600 秒 x 24 小 时 x 指定 天 数 ， 并 通过 date() 函 数 格式 化 为 指定 日 期 格式 。 
@ parent.mainFrame.location.href='release_content.php': 刷新 父 框 架 页 release_content.php 中 的 信息 。 


1.12 ”付费 信息 管理 模块 设计 


1.12.1 付费 信息 管理 模块 概述 


付费 信息 管理 模块 主要 包括 付费 信息 列表 、 付 费 信 息 审核 和 付费 信息 删除 等 3 个 功能 , 其 框架 如 图 1.69 
所 示 。 


付 织 信息 管理 
9 
付 各 信息 列表 付费 信息 市 核 付费 信息 删除 


y 
(结束 ) [在 前 台 显示 
图 1.69 付费 信息 管理 模块 的 框架 图 
1.12.2 ”付费 信息 管理 模块 技术 分 析 
付费 信息 管理 页 面 在 实现 信息 审核 及 删除 的 功能 时 应 用 到 了 UPDATE 更 新 语句 和 DELETE 删除 语 
句 。 下 面 对 这 两 个 语句 进行 详细 的 讲解 。 
1. UPDATE 语句 


UPDATE 语句 用 来 改变 单行 上 的 一 列 或 多 列 的 值 , 或 者 改变 单个 表 中 选 定 的 一 些 行 上 的 多 个 列 值 。 
UPDATE 语句 的 语法 如 下 : 


UPDATE<table_name | view_name> 
SET <column_name>=<expression> 

[.…,<last column_name>=<last expression>] 
[WHERE<search_condition>] 


UPDATE 语句 的 参数 说 明 如 表 1.10 所 示 。 
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表 1.10 UPDATE 语句 的 参数 说 明 


参 。 数 说 有明 
ee 需要 更 新 的 表 的 名 称 。 如 果 该 表 不 在 当前 服务 器 或 数据 库 中 ， 或 不 为 当前 用 户 所 有 ， 这 个 名 称 
四 可 用 链接 服务 器 、 数 据 库 和 所 有 者 名 称 来 限定 
| 要 更 新 的 视图 的 名 称 。 通 过 view_name 来 引用 的 视图 必须 是 可 更 新 的 。 用 UPDATE 语句 进行 的 
ee 修改 ， 至 多 只 能 影响 视图 的 FROM 子 句 所 引用 的 基 表 中 的 一 个 
SET 指定 要 更 新 的 列 或 变量 名 称 的 列表 


column_name 


含有 要 更 改 数据 的 列 的 名 称 。column_name 必须 驻 留 于 UPDATE 子 句 中 所 指定 的 表 或 视图 中 。 
标识 列 不 能 进行 更 新 。 如 果 指 定 了 限定 的 列 名 称 ， 限 定 符 必 须 同 UPDATE 子 句 中 的 表 或 视图 的 
名 称 相 匹 配 

变量 、 字 面值 、 表 达 式 或 加 上 括号 返回 单个 值 的 subSELECT 语句 。expression 返回 的 值 将 替换 


Re column_ name 或 @variable 中 的 现 有 值 
指定 条 件 来 限定 所 更 新 的 行 。 根 据 所 使 用 的 WHERE 子 句 的 形式 ， 有 两 种 更 新 形式 ， 
WHERE 搜索 更 新 指定 搜索 条 件 来 限定 要 删除 的 行 


<search_condition> 


定位 更 新 使 用 CURRENT OF 子 句 指定 游标 。 更 新 操作 发 生 在 游标 的 当前 位 置 
为 要 更 新 行 指定 需 满足 的 条 件 。 搜 索 条 件 也 可 以 是 联接 所 基于 的 条 件 。 对 搜索 条 件 中 可 以 包含 
的 谓词 数量 没有 限制 


ER 


下 面 应 用 UPDATE 语句 将 指定 职员 的 工资 进行 调整 ， 例 如 ， 将 “小 璇 ”的 基本 工资 由 2600 元 修 
改 为 3000 元 。 其 SQL 语句 如 下 : 


update tab_laborage set jbgz=3000 where name=' 小 璇 ' 


2. DELETE 语句 
DELETE 语句 实现 删除 数据 记录 。DELETE 语句 的 语法 如 下 : 


DELETE FROM <table_name > 
[WHERE<search-condition>] 


DELETE 语句 的 参数 说 明 如 表 1.11 所 示 。 


参数 


表 1.11 DELETE 语句 的 参数 说 明 
说 阴 


FROM 


table name 


可 选 的 关键 字 , 可 用 在 DELETE 关键 字 与 目标 table_name、view_name 或 rowset_function_limited 
之 间 


是 要 从 其 中 删除 行 的 表 的 名 称 


<search_condition> | 指定 删除 行 的 限定 条 件 。 对 搜索 条 件 中 可 以 包含 的 谓词 数量 没有 限制 


下 面 应 用 DELETE 语句 删除 “职员 姓名 = 小 璇 ”的 员工 基本 信息 。 其 SQL 语句 如 下 : 


@_ 
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DELETE tab_staffer WHERE ygname=' 小 璇 ' 


1.12.3 ”付费 信息 显示 的 实现 过 程 


国 


付费 信息 显示 页 面 使 用 的 数据 表 : tb_leaguerinfo 


管理 员 在 后 台 功能 导航 区 的 “付费 信息 显示 方式 ” 栏 中 选择 相应 的 信息 类 别 ， 然 后 按 “ 已 付费 ” 
“未 付费 ”或 “全 部 ”中 的 任意 一 种 状态 对 付费 信息 进行 管理 。 例 如 ， 在 “信息 类 别 ” 下 拉 列 表 框 中 
选择 “公寓 信息 ”， 在 “付费 状态 ”选项 组 中 选 


中 “未 付费 ” 单 选 按钮 ， 单 击 “ 检 索 ” 按 钮 提 
交 表 单 ， 程 序 将 按 指定 条 件 显示 出 符合 条 件 的 
所 有 信息 ， 运 行 结果 如 图 1.70 所 示 。 

本 系统 提供 了 一 组 单 选 按钮 组 成 的 “付费 
状态 ”选项 组 ， 分 为 已 付费 、 未 付费 和 全 部 3 
个 选项 。 选 中 “未 付费 ” 单 选 按钮 ， 则 传递 的 
值 为 0; 选中 “已 付费 ” 单 选 按钮 ， 则 传递 的 


值 为 1， 选 中 “全 部 ” 单 选 按钮 ， 则 传递 的 值 | ”| 和 ee nn 
为 all。 还 提供 了 一 个 下 拉 列 表 框 ， 供 用 户 选择 人 


信息 类 别 。 将 这 些 单 选 按钮 与 下 拉 列 表 框 都 在 | Fe 
一 个 表单 中 实现 ， 这 样 ， 当 单 击 “ 检 索 ” 按 钮 
提交 表单 后 , 选择 的 状态 会 通过 表单 进行 传递 。 


了 7 二: 未 后 息 网 后 台 管 理 系统 


I nm I 


NPEEI eb 


其 表单 代码 如 下 : 图 1.70 付费 信息 显示 页 面 的 运行 结果 


倒 程 15 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\left.php 


<form name="form3" method="post" action="find_mianfei.php" target="mainFrame"> 
<tr> 
<td height="65" align="center> 
<fieldset style="height:60;width:210"> 
<legend> 太 审核 状态 </legend> 
<input name="state" type="radio" class="input1" value="1"> 已 审核 
<input name="state" type="radio" class="input1" value="0" checked> 未 审核 
<input name="state" type="radio" class="input1" value="all"> 全 部 
</fieldset> 
</td> 
</tr> 
<tr> 
<td height="34" align="center" background="images/info_d.gif*> 信 息 类 别 : 
<select name="type"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 </option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 </option> 


</select> 
<input type="submit" name="Submit" value=" 检 索 "> 
</td> 
</tr> 
</form> 
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二 旬 代码 贴 十 

@ target: 指定 链接 的 目标 窗口 ，mainFrame 为 内 容 显 示 区 的 框架 名 称 。 另外，target 有 4 个 选项 值 ， 分 别 介绍 如 下 。 

_blank: 指定 将 链接 的 目标 文件 加 载 到 未 命名 的 新 浏览 器 窗口 中 。 

_parent: 指定 将 链接 的 目标 文件 加 载 到 包含 链接 的 父 框架 页 或 窗口 中 ， 如 果 和 包含 链接 的 框 不 是 嵌 套 的 ， 则 链 

接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 。 

_self: 指定 将 链接 的 目标 文件 加 载 到 链接 所 在 的 同一 框架 或 窗口 中 。 

_top: 指定 将 链接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 ， 并 由 此 删除 所 有 框架 。 

四 <fieldset><legend>…</legend></fieldset> 标 签 : 在 字符 集 包含 的 文本 和 其 他 元 素 外 面 绘制 一 个 方 框 。 该 元 素 是 块 
元 素 ， 必 须 成 对 出 现 。 需要 注意 的 是 ，fieldset 必须 用 在 form 表单 中 ， 一 个 表单 可 以 有 多 个 <fieldset>…</fieldset>， 每 对 
<fieldset>…</fieldset> 为 一 组 ， 每 组 的 内 容 描 述 使 用 <legend> 设 置 标题 名 称 .。 


提交 表单 信息 到 find_mianfei.php 页 ,程序 将 按 管 理 员 选择 的 指定 条 件 显示 出 符合 条 件 的 所 有 信息 。 
如 果 管 理 员 选中 “全 部 ” 单 选 按 钮 ， 那 么 系统 的 代码 如 下 : 
倒 程 16 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\find_fufei.php 


<?php 

include("../conn/conn.php"); /连接 数据 库 文件 
$state=$_POST[payfor]; // 获 取 管 理 员 选 择 的 付费 状态 
$type=$_POST[select]; // 获 取 管 理 员 选 择 的 信息 类 别 


// 如 果 管理 员 选 择 的 付费 状态 为 “全 部 ” 则 查询 管理 员 指 定 信息 类 别 下 的 所 有 付费 信息 ， 并 按 id 降序 排列 
if($state=="all"}{ 
$sql=mysql_query("select* from tb_leaguerinfo where type='$type' order by id"); 
中 
else{ 
// 如 果 管 理 员 选 择 的 付费 状态 为 “已 付费 ”或 “未 付费 ”状态 ， 则 按 管理 员 指 定 的 条 件 进行 查询 ， 并 按 id 降序 排列 
$sql=mysql_query("select* from tb_leaguerinfo where type='$type' and checkstate=$state order by id"); 


} 
$info=mysql_fetch_array($sql); /执行 SQL 语句 
?> 
， // 省 略 供求 信息 标题 的 HTML 代码 部 分 
<?php 
if($infoX{ // 如 果 检 索 到 了 查询 记录 
dof / 则 应 用 do…while 循环 语句 输出 付费 信息 
if($info[checkstate]==1){ // 如 果 付 费 状态 的 值 为 1 
$state1=" 已 付费 "; // 则 将 “已 付费 ” 赋 给 变量 $state1 
}elsef // 如 果 付 费 状态 的 值 为 0 
$state1=" 未 付费 "; // 则 将 “未 付费 ” 赋 给 变量 $state1 
?> l 
< 输出 指定 符合 查询 条 件 的 付费 信息 -一 一 一 一 一 一 一 一 一 一 一- --: > 
<tr bgcolor="#FFFFFF"> 


<td>&nbsp;<?php echo S$infoltitle];?></td> 

<td width="204">&nbsp;<?php echo S$info[content];?></td> 
<td>&nbsp;<?php echo $infollinkman];?></td> 
<td>&nbsp;<?php echo S$infoltel];?></td> 
<td>&nbsp;<?php echo S$info[sdate];?></td> 
<td>&nbsp;<?php echo $info[showday];?></td> 


局 


音 
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<td align="center" class="style11"><?php echo $state1;?></td> 

<td align="center" bgcolor="#FFFFFF"> 
<a href="statefu_ok.php?id=<?php echo Sinfolid];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
审核 </a> 
/<a href="fudel_ok.php?id=<?php echo Sinfo[id];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
删除 </a> 
</td> 
</tr> 
一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 --: > 
<?php 

}while($info=mysql_fetch_array($sql)); /do…while 循环 语句 结束 

/所 条 件 语句 结束 


// 如 果 未 检索 到 与 管理 员 指 定 条 件 相 匹配 的 记录 ， 则 输出 相关 的 提示 信息 
else{ 
?> 
<tr align="center" bgcolor="#FFFFFF"> 
<td colspan="8"> 对 不 起 ， 您 检索 的 信息 不 存在 ! </td> 

</tr> 
<?php 
$ 


?> 


4 
Pp 


1.12.4 ”付费 信息 审核 的 实现 过 程 


国 ”付费 信息 审核 使 用 的 数据 表 : tb_leaguerinfo 

经 过 审核 的 信息 说 明 该 信息 为 已 付款 信息 。 如 果 企 业 或 个 人 用 户 已 登录 供求 信息 但 未 直接 付费 ， 
想 要 后 期 付款 ， 那 么 管理 员 进 行 审核 时 ， 信 息 不 能 通过 ， 要 求 必须 进行 付款 ， 经 过 审核 的 信息 才能 在 
前 台 进 行 显示 。“ 审 核 ” 超 链接 的 代码 如 下 : 

倒 程 17 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\find_fufei.php 

hp echo Sinfolid];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 

</a> 

管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链 接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 
递 到 数据 处 理 页 statefu_ok.php， 用 UPDATE 语句 将 付费 状态 设置 为 1， 说 明 该 信息 已 经 付款 。 数 据 处 
理 页 的 代码 如 下 : 

倒 程 18 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\statefu_ok.php 


<?php 

include("../conn/conn.php"); /连接 数据 库 文件 
$id=$_GET[id]; // 获 取信 息 id 的 值 
$type=$_GETI[type]; // 获 取信 息 类 型 
$state=$_GET[state]; // 获 取信 息 付费 状态 


RC) 
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@ $sql=mysql_query("update tb_leaguerinfo set checkstate=1 where id=$id");  // 更 新 对 应 付费 信息 的 状态 


为 已 付款 
if($sq){ // 如 果 更 新 操作 成 功 
@ ”echo "<script>alert( 该 信息 已 经 通过 审核 ! 
');window.location.href="find_fufei.php?type=$type& state=$state';</script>"; 1/ 弹出 操作 成 功 提示 信息 
} 
else{ // 如 果 更 新 操作 失败 
echo "<script>alert( 该 信息 审核 操作 失败 ! ');history.back();</script>"; /1/ 弹 出 操作 失败 信息 
站 
4 代码 贴 十 


@ update.…set: 用 来 修改 指定 表 中 的 数据 。UPDATE 语句 的 使 用 方法 参见 1.12.2 节 。 

@ type=$type&state=$state: 将 变量 type 与 state 的 值 重 新 传 到 find_fufei.php 页 ， 目 的 是 为 了 使 find_fufei.php 页 中 
的 $type 和 $state 变量 重新 获得 信息 类 型 和 付费 状态 值 (管理 员 选 择 的 检索 条 件 )， 从 而 返回 到 付费 信息 管理 页 ， 并 更 新 
数据 信息 。 如 果 数据 处 理 页 不 对 这 两 个 变量 进行 传 值 ， 那 么 在 返回 到 付费 信息 管理 页 find fufei.php 时 将 会 因为 检索 不 
到 变量 的 值 而 出 错 。 


1.12.5 ”付费 信息 删除 的 实现 过 程 


国 。 付费 信息 删除 使 用 的 数据 表 : tb_leaguerinfo 
付费 信息 管理 页 中 “删除 ” 超 链 接 的 代码 如 下 : 
倒 程 19 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\find_fufei.php 


<a href="fudel_ok.php?id=<?php echo Sinfo[id];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
删除 </a> 


管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链 接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 
递 到 数据 处 理 页 fudel_ok.php， 用 DELETE 语句 将 id 指定 的 供求 信息 删除 。 数 据 处 理 页 的 代码 如 下 : 
倒 程 20 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\fudel_ok.php 


<?php 

include("../conn/conn.php"); /连接 数据 库 文件 

$id=$_GET[id]; // 获 取信 息 id 的 值 
$type=$_GETI[type]; // 获 取信 息 类 型 

$state=$_GET[state]; // 获 取信 息 付费 状态 
$sql=mysql_query("delete from tb_leaguerinfo where id=$id"); 1/ 删除 对 应 的 供求 信息 

if($sql){ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 
echo "<script>alert( 该 信息 已 经 删除 ! ')windowlocation.href=find_fufei.php?type=$type&state=$state'</script>"; 

} 

else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert(' 该 信息 删除 操作 失败 ! ');history.back();</script>"; 

} 

?> 


Or 
find_fufei.jphp， 目 的 是 返回 到 付费 信息 管理 页 ， 查 看 执行 删除 操作 后 的 状态 。 


@ 


音 
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1.12.6 ”单元 测试 


在 开发 完 管理 员 模块 后 ， 对 该 模块 进行 单元 测试 。 当 管理 员 对 付费 供求 信息 进行 审核 后 ， 审 核 操 
作成 功 了 ， 但 却 弹 出 如 图 1.71 所 示 的 错误 提示 。 


人 


图 1.71 审核 付费 供求 信息 的 错误 提示 


在 图 1.71 中 的 错误 提示 中 可 以 看 出 ， 在 付费 信息 管理 页 的 第 16 行 和 第 17 行 出 现 问题 。 下 面 看 一 
下 出 现 问题 的 这 两 行 代码 ; 


if($state=="all")f /如果 管理 员 选 择 的 付费 状态 为 “全 部 ” 则 执行 下 面 的 SQL 语句 
$sql1=mysql_query("select count(*) as total from tb_leaguerinfo ”where type='$type' order by id"); 
}elsef /否则 ， 执 行 下 面 的 SQL 语句 
$sql1=mysql_query("select count(*) as total from tb_leaguerinfo where type='$type' and checkstate=$state 
order by id"); 


人 


$minfo=mysql_fetch_array($sql1); 


人 


$total=$minfo[total]; 


从 代码 中 可 以 看 出 ，SQL 语句 的 书写 并 没有 错误 。 根 据 图 1.71 所 示 页 面 的 结果 ， 当 前 信息 类 别 为 
空 ， 则 说 明 管 理 员 选择 的 信息 类 别 没有 传 过 来 值 。 由 此 可 以 看 出 ， 这 是 由 于 在 执行 审核 后 页 面 重 新 刷 
新 了 ， 因 此 检索 不 到 管理 员 选 定 的 查询 条 件 值 。 
“审核 ” 超 链接 的 源 代码 如 下 : 
倒 程 21 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\find_fufei.php 
<a href="statefu_ok.php?id=<?php echo $info[id];?>"> 审 核 </a> 


审核 处 理 页 的 源 代码 如 下 : 
倒 程 22 ”代码 位 置 : 光盘 \TM\01\99pursey\admin\statefu_ok.php 

<?php 

$id=$_GET[d]:; /获取 信息 id 的 值 
$sql=mysql_query("update tb_leaguerinfo set checkstate=1 where id=$id");，// 更 新 对 应 付费 信息 的 状态 为 已 付款 
if($sql){ // 如 果 更 新 操作 成 功 ， 弹 出 提示 信息 


echo "<script>alert(' 该 信息 已 经 通过 审核 ! ');window.location.href="find_fufei.php;</script>"; 
上 


@ 
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解决 该 问题 的 方法 需要 在 “审核 ” 超 链接 传 值 时 将 管理 员 选 定 的 “信息 类 型 ”和 “审核 状态 ”的 
变量 值 一 同 传递 到 数据 处 理 页 ， 当 审核 操作 完成 后 ， 再 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 重 新 
传递 给 付费 信息 管理 页 find_fufei.php 即 可 。 

“审核 ” 超 链接 修改 后 的 代码 〈 加 粗 的 代码 部 分 为 修改 的 代码 部 分 ) 如 下 : 

倒 程 23 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\find_fufei.php 


<a href="statefu_ok.php?id=<?php echo Sinfo[id];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
审核 </a> 


区 技 西 在 可 多 个 赤 量 时 ， 灾 重 之 间 用 “&。 符号 分 村 

管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传 
递 到 数据 处 理 页 statefu_ok.php。 在 执行 完 更 新 操作 后 ， 需 要 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 
重新 传递 到 付费 信息 管理 页 find_fufei.php， 加 粗 的 代码 部 分 为 修改 的 代码 部 分 。 

倒 各 24 ”代码 位 置 ， 光盘 \TM\01\99pursey\admin\statefu_ok.php 


<?php 
$id=$_GETI[id]; 1/ 获取 信息 id 的 值 
$type=$_GETI[type]; // 获 取信 息 类 型 
$state=$_GET[state]; // 获 取信 息 付费 状态 
$sql=mysql_query("update tb_leaguerinfo set checkstate=1 where id=$id"); // 更 新 对 应 付费 信息 的 状态 为 已 付款 
if($sql){ // 如 果 更 新 操作 成 功 

echo "<script>alert(' 该 信息 已 经 通过 审核 ! ');window.location.href='find_fufei.php? type=$type&state= 
$state';</script>"; // 弹 出 操作 成 功 提示 信息 
} 


1.13 网 站 发 布 


开发 九 九 度 供求 信息 网 的 最 终 目 的 是 将 其 发 布 到 Internet 上 ， 供 用 户 浏览 访问 。 

在 服务 器 上 上 载 网 站 ， 首 先 需 从 服务 商 处 申请 固定 的 卫 号 ， 然 后 再 注册 一 个 域名 ， 并 将 域名 指定 
到 该 IP 地 址 。 在 服务 器 上 安装 PHP 的 开发 环境 ， 最 后 上 载 网 站 。 

下 面 以 九 九 度 供求 信息 网 为 例 ， 来 讲解 在 个 人 服务 器 上 上 载 网 站 的 过 程 。 

(1) 右 击 “网 上 邻居 ” 在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ， 打 开 “ 网 络 连接 ”对 话 框 。 

(2) 右 击 “本 地 连接 ”， 在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ,打开 “本 地 连接 属性 ”对 话 框 ， 
如 图 1.72 所 示 。 在 该 对 话 框 的 “此 连接 使 用 下 列 选 定 的 组 件 ” 列 表 框 中 选中 “Internet 协议 (TCP/IP)” 
复 选 框 ， 然 后 单 击 “ 属 性 ”按钮 ， 按 服务 商 提供 的 指定 IP 设置 服务 器 的 卫 地 址 ， 如 图 1.73 所 示 。 

(3) 安装 PHP 的 开发 环境 ， 参 见 本 章 1.4 节 。 

(4) 将 九 九 度 供求 信息 网 上 载 到 服务 器 指定 的 路 径 下 ， 如 图 1.74 所 示 。 

(5) 在 正 浏 览 器 地 址 栏 中 输入 申请 的 域名 ， 发 布 的 网 站 运行 结果 如 图 1.75 所 示 。 


@_ 
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图 1.74 上 载 九 九 度 供求 信息 网 到 指定 的 路 径 下 图 1.75 九 九 度 供求 信息 网 发 布 后 的 运行 结果 
1.14 开发 技巧 与 难点 分 析 
1.14.1 查询 关键 字 描 红 功能 
在 九 九 度 供求 信息 网 前 台 信息 检索 过 程 中 体现 了 方便 快捷 的 人 性 
化 原则 ， 为 了 方便 浏览 者 查阅 信息 ， 便 于 查找 与 浏览 者 的 关键 字 相符 A 
合 的 信息 ， 在 搜索 引擎 中 添加 描 红 功能 。 
查询 关键 字 描 红 是 指 将 查询 关键 字 以 特殊 的 颜色 、 字 号 或 字体 进 用 户 
行 标识 ， 这 样 可 以 使 浏览 者 快速 找到 所 需 的 关键 字 ， 方 便 浏览 者 从 搜 
索 结果 中 查找 所 需 内 容 。 查 询 关键 字 描 红 适用 于 模糊 查询 。 下 面 介绍 设置 查询 条 件 
如 何 实现 查询 关键 字 描 红 。 
本 系统 用 str_ireplace() 函 数 来 蔡 换 查 询 关键 字 , 当 显示 所 查询 的 相 本 
关 信息 时 ， 将 输出 的 关键 字 的 字体 符 换 为 红色 。 描 红 功能 的 业务 流程 
如 图 1.76 所 示 。 
str_ireplace0 函 数 用 于 将 某 个 指定 的 字符 串 蔡 换 为 另 一 个 指定 的 人 


字符 串 ， 不 区 分 大 小 写 。 该 函数 的 语法 如 下 : 


1.76 “ 描 红 功能 的 业务 流程 


@ 
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mixed str_ireplace ( mixed search, mixed replace, mixed subject [, int &count]) 


该 函数 将 所 有 在 参数 subject 中 出 现 的 参数 search 以 参数 replace 取代 ， 参 数 &count 表示 取代 字符 
串 执行 的 次 数 。 
str_ireplace0 函 数 的 参数 说 明 如 表 1.12 所 示 。 
表 1.12 str_ireplace() 函 数 的 参数 说 明 


参 。 数 说 有明 
search | 必要 参数 ， 指 定 需 要 查找 的 字符 串 
replace | 必要 参数 ， 指 定 蔡 换 的 值 
subject | 必要 参数 ， 指 定 查找 的 范围 


count 可 选 参 数 ， 获 取 执 行 蔡 换 的 数量 


orn a En RR GR 
用 str_replace(O) 函 数 。 


本 系统 应 用 str_ireplace0) 函 数 替 换 查询 字符 串 为 红色 的 字符 串 ， 关 键 代 码 如 下 : 


<?php 
include("conn/conn.php"); /| 连接 数据 库 文件 
$content=$_POST[content]; // 获 取 查 询 关 键 字 


$sql1=mysql_query("select * from tb leaguerinfo where checkstate=1 and type='$type' and content 
like'%$content%' or title like'%$content%' or linkman like'%$content%' or tel like'% $content%"); 
$info1=mysql_fetch_array($sql1); // 采 用 模糊 信息 资源 查询 
// 下 面 应 用 str_ireplace() 函 数 将 指定 的 查询 关键 字 用 红色 文字 替代 ， 并 输出 蔡 换 后 的 字符 串 

echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>", $info1[type]);// 替 换 信息 类 型 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>", $info1[title]); /替换 信息 标题 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[edate]); /替换 发 布 时 间 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[content]); /替换 信息 内 容 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000>".$content."</font>",$info1llinkman]); /替换 联系 人 为 红色 字体 
echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1[tel]); // 蔡 换 联系 电话 为 红色 字体 
?> 


1.14.2 ”表单 数据 的 提交 方式 


获取 表单 元 素 提交 的 值 是 表单 应 用 中 最 基本 的 操作 方法 。 表 单数 据 的 传送 方法 有 get 方法 和 post 
方法 两 种 , 通过 <form> 的 method 属性 来 指定 。 下 面 来 具体 讲解 一 下 这 两 种 方法 在 实际 工作 中 的 应 用 范 
围 和 使 用 技巧 。 

1. 通过 get 方 法 提交 数据 

使 用 get 方法 时 ， 表 单数 据 被 当 作 url 的 一 部 分 一 起 传 过 去 。 格 式 如 下 : 

http://url?name1=value1&name2=value2... 


url: 表单 响应 地 址 。 例 如 ，127.0.0.1/index.php。 


局 
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name: 表单 元 素 的 名 称 。 例 如 , <input type="text" name="user">, 这 里 name 的 属性 值 就 是 user。 
通过 name 值 可 以 获取 value 的 属性 值 。 

value: 表单 元 素 的 值 。 例 如 ，<input type="text" name="user" value="mr">， 意 思 是 名 字 叫 user 
的 text 表单 元 素 的 值 为 mr。 


说明 url 和 表单 元 素 之 间 用 “2?” 隔 开 ， 而 多 个 表单 元 素 之 间 用 “&” 隔 开 ， 每 个 表单 元 素 的 格 
式 都 是 “name=value”， 固 定 不 变 。 
PHP 使 用 $_GET 预定 义 变量 自动 保存 通过 get 方 法 传 过 来 的 值 ， 使 用 格式 为 : 
$_GETIname] 


这 样 ， 就 可 以 直接 使 用 名 字 为 name 的 表单 元 素 的 值 。 
加 有 的 PHP 版 本 中 直接 写 Sname 就 可 以 调用 表单 元 素 的 值 ,这 和 phpini 文 件 的 配置 有 关系 ， 
定位 到 GLOBAL=ON/OFF 行 ， 如 果 值 为 ON， 就 可 以 直接 写成 Sname， 反 之 则 不 可 以 。 直 接应 用 表 
单 名 称 十 分 方便 ， 但 也 存在 一 定 的 安全 隐患 。 推 荐 读者 关闭 GLOBAL 项 。 


使 用 文本 框 传 值 的 程序 中 包含 一 个 文本 框 元 素 。 在 文本 框 中 输入 信息 ， 当 单 击 “提交 ”按钮 时 ， 
文本 框 内 的 信息 就 会 和 url 一 起 显示 在 地 址 栏 中 。 代 码 如 下 : 


<form name="login" method="get" action="index.php"> 


</form> 


| 


[LS 技巧 get 方 法 是 使 用 url 来 进行 传 值 的 ,“ 加 入 收藏 夹 ”的 功能 就 是 get 方 法 的 应 用 之 一 。 想 要 
将 一 个 带 参 数 的 网 址 加 入 到 收藏 夹 中 ， 只 能 使 用 url， 它 可 以 脱离 表单 的 束 绑 。 文字、 图 片 等 都 可 
以 使 用 这 种 方法 来 传 值 。 


2. 通过 post 方法 提交 数据 

get 方法 有 个 最 大 的 缺点 ， 就 是 它 的 信息 是 显示 在 客户 端 浏览 器 上 的 ， 这 使 用 户 的 资料 暴露 无 遗 ， 
而 且 url 本 身受 长 度 限制 (1024KB ), 不 能 传输 较 大 的 数据 。 这 时 可 以 选择 post 方法 。 使 用 时 , 将 <form> 
表单 中 的 属性 method 设置 成 post 即 可 。post 方法 不 依赖 url， 所 有 提交 的 信息 在 后 台 传输 ， 不 会 显示 
在 地 址 栏 中 ， 安 全 性 高 ， 而 且 没 有 长 度 限 制 。 

使 用 PHP 的 $_POST[name] 变 量 可 以 获取 表单 元 素 的 值 ， 格 式 和 $_GET[name] 类 似 : 


$_POSTIname] 
例如 ， 使 用 的 post 方 法 返回 文本 框 信息 ， 代 码 如 下 : 
<form name="|login" method="post" action="index.php"> 


</form> 
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1.15 MySQL 数据 库 技术 专题 


phpMyAdmin 是 众多 MySQL 图 形 化 管理 工具 中 应 用 最 广泛 的 一 种 ， 它 是 基于 PHP 语言 编写 的 ， 
该 工具 是 B/S 结构 、 基 于 Web 跨 平 台 的 管理 程序 ， 并 且 支 持 简 体 中 文 ， 可 以 在 官方 网 站 免费 下 载 。 安 
装 后 在 浏览 器 地 址 栏 中 输入 http://127.0.0.1/phpMyAdmin/， 即 可 进入 MySQL 的 管理 界面 。 

phpMyAdmin 为 Web 开发 人 员 提 供 了 类 似 于 Access、SQL Server 的 图 形 化 数据 库 操作 界面 ， 通 过 
该 管理 工具 可 以 进行 绝 大 部 分 的 MySQL 操作 ， 包 括 对 数据 库 及 数据 表 的 建立 和 维护 。 


1.15.1 创建 和 删除 数据 库 


1. 创建 数据 库 


在 phpMyAdmin 的 主 界面 中 有 两 个 文本 框 和 “创建 ”按钮 ， 首 先 在 文本 框 中 输入 数据 库 的 名 称 ， 
然后 选择 编码 ， 最 后 单 击 “ 创 建 ”按钮 ， 这 样 新 的 数据 库 就 可 以 被 创建 成 功 。 例 如 ， 创 建 一 个 名 称 为 
db_pursey 的 数据 库 ， 首 先 在 文本 框 中 输入 db_pursey， 之 后 在 下 拉 列 表 框 中 选择 要 使 用 的 编码 ， 在 
Windows 下 一 般 选择 gb2312_chinese_ci， 如 图 1.77 所 示 。 最 后 单 击 “ 创 建 ”按钮 ， 这 样 名 为 db_pursey 
的 数据 库 就 被 创建 成 功 ， 执 行 结果 如 图 1.78 所 示 。 

该 数据 库 名 称 出 现在 左 侧 导航 栏 的 数据 库 下 拉 菜 单 中 ， 选 择 这 个 数据 库 ， 在 右 侧 界面 中 可 以 对 该 
数据 库 进 行 操作 ， 如 结构 、SQL、 导 出 、 搜 索 、 查 询 、 删 除 等 ， 单 击 相应 的 按钮 即 可 进入 相应 的 操作 
界面 。 但 是 在 创建 数据 库 还 没有 创建 数据 表 的 情况 下 ， 只 能 够 执行 结构 、SQL 和 删除 3 项 操作 ， 其 他 
操作 不 可 以 执行 。 


因 记 回避 | 加 划 CREATE DATAB 


图 1.77 phpMyAdmin 管理 界面 图 1.78 数据 库 的 建立 
2. 删除 数据 库 


要 删除 某 个 数据 库 ， 首 先 在 左 侧 的 下 拉 菜 单 中 选择 该 数据 库 ， 然 后 单 击 右 侧 界面 中 的 “删除 ” 按 
钮 即 可 。 


1.15.2 ”创建 和 删除 数据 表 


针对 表 级 操作 是 在 选 定 了 数据 库 的 情况 下 进行 的 , 即 表 级 操作 的 前 提 是 用 户 必须 选择 一 个 数据 库 ， 
在 该 数据 库 中 进行 表 的 建立 和 维护 。 


@ 
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1. 创建 数据 表 


创建 数据 库 db_pursey 后 ， 在 右 侧 页 面 中 会 出 现 如 图 1.78 所 示 的 数据 表 创建 提示 页 面 ， 完 成 数据 
表 的 创建 操作 。 

首先 在 表单 中 输入 数据 表 的 名 称 和 字段 数 ， 然 后 单 击 “执行 ”按钮 ， 进 入 各 个 字段 的 详细 信息 录 
入 表单 ， 包 括 字段 名 、 数 据 类 型 、 长 度 / 值 、 属 性 、 默 认 值 、 额 外 和 索引 的 类 型 等 ， 在 这 里 就 完成 了 对 
表 结 构 的 详细 设置 ， 如 图 1.79 所 示 。 

当 所 有 的 信息 都 输入 完 以 后 ， 就 可 以 单 击 “ 保 存 ” 按 钮 ， 成 功 创建 数据 表 tb_admin。 一 个 新 的 数 
据 表 被 创建 后 ， 进 入 到 数据 表 页 面 ， 在 这 里 可 以 通过 改变 表 的 结构 来 修改 表 ， 可 以 执行 添加 新 的 列 、 
删除 列 、 索 引 列 、 修 改 列 的 数据 类 型 或 者 字段 的 长 度 / 值 等 操作 ， 如 图 1.80 所 示 。 


园 服务 器 :localhost ， 轧 数据 库 ; db_pursey ， 国 表 :tb_admin 
a st2 


明 服务 器 :eealhest ， 局 数据 内:dh_pursey ， 加 玉 BE 


er 


于 区 
Pry 1 PX 创建 索引 


ed Ed 了 及 二 | 夏天 9 奸 来 3| 执 生 
图 1.79 创建 表 信 息 图 1.80 操作 列表 
2. 删除 数据 表 


执行 删除 表 的 操作 很 简单 ， 只 要 单 击 图 1.80 所 示 页 面 上 方 的 “删除 ”按钮 ， 就 可 以 轻松 地 删除 当 
前 数据 表 。 


1.16 本 章 总 


本 章 依据 软件 开发 流程 介绍 了 九 九 度 供求 信息 网 的 开发 过 程 。 在 开发 任何 一 个 项 目前 ， 首 先 要 充 
分 做 好 前 期 准备 ， 如 完善 的 需求 分 析 、 清 晰 的 业务 流程 、 合 理 的 程序 结构 等 ， 这 样 在 后 期 的 程序 开发 
中 才 会 得 心 应 手 ， 有 备 无 患 。 通 过 本 章 的 学 习 ， 读 者 可 以 了 解数 据 库 建 模 的 概念 ， 掌 握 PowerDesigner 
数据 库 建 模 的 方法 ， 熟 悉 框架 技术 在 Web 应 用 程序 中 的 应 用 。 


(A 


全 各 < 


GE 时 


BCTY365 网 上 社区 


(Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


随 着 人 类 文明 的 不 断 进步 ， 网 络 这 个 虚拟 世界 也 在 发 生 着 变化 ， 现 
实 世 界 中 的 所 有 内 容 在 网 络 上 都 有 反映 。 现 实 世 界 中 物 以 类 聚 ， 人 以 群 
分 ， 网 络 世 界 中 也 形成 了 各 式 各 样 的 社区 。 这 种 社区 建立 在 特殊 的 兴趣 
点 《如 体育 、 新 闻 、 穹 物 、 电 影 和 游戏 等 上 特殊 的 人 群 、 特 殊 的 爱好 
或 者 是 特殊 的 服务 上 等 。 社 区 也 是 一 种 特定 的 商业 模式 ， 它 本 身 可 以 进 
行 电子 商务 ， 也 可 以 完全 与 商务 无 关 。 

所 谓 网 上 社区 是 指 包 括 BBS 论坛 、 聊 天 室 、 博 客 等 形式 在 内 的 网 上 
交流 空间 ， 同 一 主题 的 网 上 社区 集中 了 具有 共同 兴趣 的 访问 者 ， 由 于 有 
众多 用 户 的 参与 ， 因 此 具备 了 交流 的 功能 ， 成 为 一 个 营销 场所 。 

网 上 社区 有 各 种 不 同 的 表现 形式 和 规模 ， 有 个 人 创办 的 社区 ， 功 能 
和 界面 追求 时 尚 、 个 性 突出 ;) 有 大 型 的 商业 性 质 社区 ， 以 盈利 为 目的 
分 类 多 元 化 ， 适 合 不 同类 型 的 网 民 。 

本 章 开 发 的 BCTY365 网 上 社区 主要 面向 程序 开发 人 员 , 集 论 坛 . 留言 板 、 
软件 下 载 、 升 级 下 载 、 技 术 支 持 和 在 线 购物 等 功能 于 一 身 ， 既 是 一 个 程序 开 
发 者 交流 的 平台 ， 更 是 一 个 网 络 营销 的 场所 。 通 过 阅读 本 章 ， 可 以 学 习 到 


WI 网 上 社区 开发 的 基本 过 程 由， 如何 设计 公共 类 


由 如何 做 需求 分 析 和 系统 设计 MH ”软件 上 传 和 下 载 功能 的 实现 方法 
”在 Linux 操作 系统 下 搭建 PHP ”在线 论坛 功能 的 实现 方法 

开发 环境 MW 在 Linux 操作 系统 下 发 布 网 站 
只 ”如 何 设计 和 创建 数据 库 、 数 据 表 MW 在 线 支付 技术 
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2 和 汪 妇 有 每 


随 着 市 场 竞争 的 日 益 激烈 ， 企 业 的 生存 和 发 展 之 路 更 加 艰难 ， 要 想 使 企业 保持 旺盛 的 生命 力 ， 企 
业 必 须要 跟 上 时 代 发 展 的 脚步 ， 不 断 为 企业 注入 新 的 活力 。 某 科技 公司 为 适应 市 场 的 需求 ， 增 加 公司 
在 互联 网 上 的 影响 力 ， 将 开发 一 个 网 上 社区 系统 ， 为 广大 的 编程 爱好 者 提供 一 个 交流 的 平台 ， 并 且 以 
此 来 推广 该 公司 的 软件 产品 。 


2.2 系统 分 析 


当 一 个 开发 项 目 被 确立 时 ， 首 先 要 做 的 就 是 需求 分 析 、 可 行 性 分 析 ， 然 后 编写 项 目 计划 书 ， 以 使 
项 目 开发 人 员 了 解 和 掌握 网 站 的 前 期 策划 和 网 站 开发 流程 。 


2.2.1 需求 分 析 


在 开发 网 上 社区 之 前 ， 首 先 要 明确 所 要 开发 的 社区 属于 什么 类 型 ， 是 个 人 的 社区 系统 ， 还 是 商业 
化 的 社区 系统 ， 并 且 要 知道 开发 的 社区 是 面向 什么 样 的 人 群 ， 是 普通 网 民 ， 还 是 专业 的 技术 人 员 ， 或 
者 是 其 他 的 特殊 群体 。 针 对 不 同 的 人 群 ， 社 区 应 该 具有 不 同 的 特点 。 当 明确 了 这 些 ， 项 目 开 发 的 思路 
就 清晰 了 ， 然 后 再 对 网 站 上 一 些 相关 的 社区 进行 考察 、 分 析 ， 从 中 吸取 经 验 ， 并 结合 企业 的 要 求 以 及 
实际 的 市 场 调查 结果 ， 提 出 一 个 合理 的 网 上 社区 网 站 功能 架构 。 本 网 站 需求 如 下 : 
网 站 设计 页 面 要 求 整 洁 、 美 观 大 方 ， 能 够 展示 企业 形象 。 
网 站 页 面具 有 Banner 广告 ， 树 立 企 业 良 好 的 口碑 宣传 。 
设计 主要 从 编程 者 的 角度 考虑 ， 为 编程 者 解决 在 开发 中 出 现 的 问题 。 
展示 出 企业 全 力 推出 的 软件 产品 和 提供 的 免费 软件 ， 以 此 吸引 浏览 者 。 
提供 一 个 良好 的 网 上 购物 的 操作 平台 。 
提供 技术 支持 ， 解 决 编程 过 程 中 常见 的 问题 。 
提供 一 个 讨论 和 研究 问题 的 平台 。 
做 到 让 广大 浏览 者 关注 企业 的 动态 。 
为 客户 提供 反馈 信息 的 平台 ， 能 够 做 到 及 时 与 客户 进行 沟通 。 
完善 的 后 台 管 理 系统 。 


2.2.2 可 行 性 分 析 


可 行 性 分 析 是 世界 上 普遍 采用 的 一 种 研究 工程 项 目 是 否 可 行 的 科学 。 其 通过 各 种 有 效 的 方法 ， 对 
工程 项 目 进行 分 析 ， 从 技术 、 经 济 、 市 场 等 方面 加 以 评价 ， 最 终 给 投资 决策 者 提供 是 否 选择 该 项 目 进 
行 开发 的 依据 。 

BCTY365 网 上 社区 项 目 开发 的 可 行 性 分 析 主 要 从 以 下 两 个 方面 考虑 。 


加 


办 办 办 办 办 办 多多 轨 
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1. 经 济 可 行 性 分 析 

企业 为 扩大 公司 的 影响 力 ， 推 出 软件 产品 ， 采 用 网 上 社区 的 形式 在 网 络 上 进行 推广 ， 不 但 可 以 汇 
聚 更 多 的 人 气 ， 而 且 可 以 让 更 多 的 人 了 解 该 企业 ， 从 而 达到 推广 企业 软件 产品 的 目的 ， 最 终 为 企业 带 
来 更 大 的 收益 。 更 重要 的 一 点 是 采取 该 方法 的 成 本 相对 其 他 的 电视 广告 或 者 人 力 宣传 的 成 本 要 低 得 多 ， 
虽然 周期 很 长 ， 但 是 却 能 够 取得 长 期 的 收益 。 

2. 技术 可 行 性 分 析 


网 上 社区 系统 的 开发 采用 的 是 Apache+ PHP+phpMyAdmin+MySQL 5.0， 开 发 软件 都 是 免费 的 ， 可 
以 直接 从 网 上 下 载 ， 无 须 支付 任何 费用 。 要 完成 BCTY365 网 上 社区 系统 的 开发 ， 必 须 能 够 配置 PHP 
程序 开发 的 环境 ， 掌 握 在 线 支付 、 购 物 车 和 在 线 论坛 技术 。 


2.2.3 ”编写 项 目 计划 书 


根据 《GB8567 一 88 计算 机 软件 产品 开发 文件 编制 指南 》 中 的 项 目 开发 计划 要 求 ,结合 单位 实际 情 
况 ， 设 计 项 目 计划 书 如 下 。 


1. 引言 


(1) 编写 目的 

为 了 保证 项 目 开发 人 员 按 时 保质 地 完成 预订 目标 ， 更 好 地 了 解 项 目 实际 情况 ， 按 照 合 理 的 顺序 开 
展 工作 ， 现 以 书面 的 形式 将 项 目 开 发 生命 周期 中 的 项 目 任务 范围 、 项 目 团队 组 织 结构 、 团 队 成 员 的 工 
作 责 任 、 团 队 内 外 沟通 协作 方式 、 开 发 进度 、 检 查 项 目 工作 等 内 容 描述 出 来 ， 作 为 项 目 相关 人 员 之 间 
的 共识 和 约定 ， 作 为 项 目 生 命 周期 内 的 所 有 项 目 活动 的 行动 基础 。 

(2) 背景 

BCTY365 网 上 社区 系统 是 本 公司 与 X X X 信 息 科 技 有 限 公司 签 定 的 待 开发 项 目 , 网 站 性 质 为 信息 
服务 类 型 ， 为 企业 与 客户 、 浏 览 者 和 会 员 之 间 提 供 一 个 技术 交流 平台 ， 并 且 全 力 推出 企业 的 软件 产品 。 
项 目 周期 为 两 个 月 ， 项 目 背景 规划 如 表 2.1 所 示 。 


表 2.1 项 目 背景 规划 


项 目 名 称 签 定 项 目 单位 项 目 负责 人 项 目 承 担 部 门 
甲 方 ，X XX 信息 科技 有 限 公司 设计 部 门 
BCTY365 网 上 社区 系统 开发 部 门 
乙方 ，X XX 网络 科技 有 限 公司 测试 部 门 
2. 概述 
(1) 项 目 目标 
项 目 目标 应 当 符合 SMART 原则 ， 把 项 目 要 完成 的 工作 用 清晰 的 语言 描述 出 来 。BCTY365 网 上 社 
区 系统 的 项 目 目标 如 下 : 


BCTY365 网 上 社区 系统 主要 用 于 在 网 络 中 树立 企业 的 形象 ， 为 程序 开发 者 提供 一 个 交流 的 平台 ， 
拉 近 企业 与 客户 、 会 员 和 浏览 者 之 间 的 距离 ， 从 而 达到 推广 企业 软件 产品 的 目的 。 整 个 项 目 需 要 两 个 
月 结束 ， 交 给 客户 进行 验收 。 


@ 
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(2) 产品 目标 与 范围 
一 方面 BCTY365 网 上 社区 系统 能 够 为 企业 节省 大 量 人 力 资源 , 企业 不 再 需要 大 量 的 业务 人 员 去 跑 
市 场 ， 间 接 为 企业 节约 了 成 本 。 另 一 方面 ， BCTY365 网 上 社区 系统 能 够 收集 海量 编程 问题 的 解决 方案 
和 好 的 建议 ， 将 会 有 大 量 用 户 访问 网 站 ， 有 助 于 提高 企业 知名 度 。 
(3) 应 交付 成 果 
项 目 开发 完成 后 ， 交 付 的 内 容 如 下 : 
以 光盘 的 形式 交付 BCTY365 网 上 社区 系统 的 源 程序 、 网 站 数据 库 文件 、 系 统 使 用 说 明 书 。 
客户 方 应 用 自己 的 服务 器 ， 因 此 需要 乙方 架设 Apache 服务 器 ， 安 装 PHP 开发 环境 ， 协 助 甲 
方 购买 域名 ， 将 开发 的 BCTY365 网 上 社区 系统 发 布 到 互联 网 上 运行 。 
网 站 发 布 到 互联 网 上 以 后 ， 进 行 后 期 的 6 个 月 无 偿 维护 与 服务 ， 超 过 6 个 月 后 进行 网 站 有 偿 
维护 与 服务 。 
(4) 项 目 验收 方式 与 依据 
项 目 验 收 分 为 内 部 验收 和 外 部 验收 两 种 方式 。 在 项 目 开发 完成 后 ， 首 先进 行内 部 验收 ， 由 系统 测 
试 员 根据 用 户 需 求 和 项 目 目标 进行 验收 。 项 目 在 通过 内 部 验收 后 ， 交 给 用 户 进行 验收 ， 验 收 的 主要 依 
据 为 需求 规格 说 明 书 。 
3. 项 目 团队 组 织 
(1) 组 织 结构 


为 了 完成 BCTY365 网 上 社区 系统 的 项 目 开 发 ,公司 组 建 一 个 临时 的 项 目 团队 ， 由 项 目 经 理 、 系 统 
分 析 师 、PHP 开发 工程 师 、 网 页 设计 师 和 系统 测试 员 构 成 ， 如 图 2.1 所 示 。 


项 目 经 理 
素 吉本 Pe 开发 工程师 ii 师 二 
图 2.1 项 目 团队 组 织 结构 图 
《2) 大 员 分 王 
为 了 明确 项 目 团队 中 每 个 人 的 任务 分 工 ， 现 制定 人 员 分工 表 ， 如 表 2.2 所 示 。 


表 2.2 人 员 分 工 表 


姓 ”名 技术 水 平 
张 达 明 | MBA 
王 言 辉 “| 高 级 系统 分 析 师 
潘 攀 ”| 高 级 PHP 工程 师 


刘 悦 高 级 美工 设计 师 
高 级 系统 测试 工程 师 项 目 开发 系统 测试 员 


工作 描述 
负责 项 目的 审批 决策 的 实施 、 项 目的 前 期 分 析 、 
策划 、 项 目 开发 进度 的 跟踪 、 项 目 质量 的 检查 
负责 系统 功能 分 析 、 系 统 框架 设计 
负责 软件 前 后 台 设计 与 编码 
负责 网 页 风格 的 确定 、 网 页 图 片 的 设计 
对 软件 进行 测试 、 编 写 软件 测试 文档 
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2.3 系统 设计 


2.3.1 系统 目标 


根据 对 目前 网 络 上 各 种 社区 的 分 析 和 研究 ,结合 本 项 目的 实际 需求 ， 在 设计 时 应 该 满足 以 下 目标 : 
界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵活 ， 树 立 企 业 形象 。 

功能 完善 、 结 构 清晰 。 

重点 突出 企业 的 软件 产品 。 

及 时 更 新 网 站 公告 。 

及 时 查阅 和 回复 客户 反馈 信息 。 

为 用 户 提供 沟通 和 交流 的 平台 。 

购物 车 模块 的 设计 结构 合理 、 流 程 清晰 。 

购物 结算 功能 设计 符合 逻辑 ， 计 算 准 确 。 

订单 处 理 功能 的 设计 及 时 、 准 确 、 安 全 。 

处 理 好 网 上 支付 功能 的 设计 与 网 上 银行 之 间 数 据 的 传递 。 

具备 完善 的 后 台 管 理 功能 ， 能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 
系统 运行 稳定 ， 有 具备 良好 的 防范 措施 。 


系统 功能 结构 


结合 需求 分 析 和 系统 目标 中 的 内 容 ，BCTY365 网 上 社区 系统 的 功能 结构 已 经 设计 完成 。 为 了 使 读 
者 能 够 更 清楚 地 了 解 网 站 的 结构 ， 下 面 给 出 BCTY365 网 上 社区 前 台 和 后 台 功 能 模块 结构 图 。 
BCTY365 网 上 社区 前 台 管 理 系统 的 功能 设计 如 图 2.2 所 示 。 


国共 办 办 办 办 办 办 办 办 办 的 


2.3， 


De 


论坛 版 
查找 帖子 让 | 订单 处 理 |] 
而 主帅 | 开行 同上 支 和 
生子 测 虹 i 
和 现 物 结算 
友子 

版 沁 距 转 


图 2.2 网 上 社区 前 台 功能 模块 结构 图 
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BCTY365 网 上 社区 后 台 管 理 系 统 的 功能 设计 如 图 2.3 所 示 。 


添加 升级 包 


编辑 升级 包 


添加 序列 号 


编辑 序列 号 


2.3.3 ”系统 预览 


添加 编程 词典 版 本 


编辑 编程 词典 版 本 


添加 编程 词典 


编辑 编程 词典 


和 


图 2.3 网 上 社区 后 台 功能 模块 结构 图 


BCTY365 网 上 社区 系统 由 多 个 程序 页 面 组 成 ， 下 面 给 出 儿 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 


程序 。 


前 台 首 页 如 图 2.4 所 示 ,该 页 面 用 于 展示 本 系统 的 功能 模块 ,突出 企业 的 形象 , 推广 企业 的 软件 产 
品 。 后 台 首 页 如 图 2.5 所 示 ， 该 页 面 用 于 实现 对 编程 词典 、 技 术 支 持 、 软 件 升 级 、 软 件 试用 等 内 容 的 


管理 。 


图 2.4 前 台 首 页 (光盘 \TM\02\bcty365\index.php) 


0: yet-asnanl Comen sm C1 Bird 


图 2.5 后 台 首页 (光盘 \TM\02\bcty365\ admin\default.php) 


在 线 订购 模块 的 页 面 效果 如 图 2.6 所 示 , 该 页 面 主要 用 于 展示 本 企业 在 线 推出 的 软件 产品 , 实现 对 


@ 
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产品 的 在 线 购买 功能 。 软 件 下 载 模块 的 页 面 效 果 如 图 2.7 所 示 , 该 页 面 主要 用 于 展示 本 企业 提供 的 免费 
软件 ， 并 且 提 供 下 载 链接 。 


BCTY365 社 区 网 ， 找 你 想 要 的 


亲 间 ， 基 四 厂 的 避 久 什 训 


图 2.6 在 线 订 购 ( 光 盘 \TM\02\bcty365\ morebccd.php) 图 2.7 软件 下 载 (光盘 \TM\02\bcty365\ rjxz.php) 


社区 论坛 模块 的 页 面 效果 如 图 2.8 所 示 , 该 页 面 主要 用 于 展示 论坛 中 的 各 大 版 块 , 并 且 提 供 超 链接 
跳 转 到 对 应 的 版 块 。 后 台 的 登录 页 面 效果 如 图 2.9 所 示 ， 该 页 面 主要 实现 后 台 管理 员 登 录 。 


图 2.8 社区 论坛 (光盘 \TM\02\bcty365\bbs_index.php) ”图 2.9 后 台 登 录 (光盘 \TMW2\bcty36S\admin\index.php) 
2.3.4 开发 环境 


在 开发 BCTY365 网 上 社区 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows Server 2003/Linux (推荐 )。 
服务 器 : Apache 2.2.8。 

PHP 软件 : PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.10.3。 
开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辨 率 : 最 佳 效果 为 1024x768 像素 。 


但 
因 因 因 因 办 因 办 加 
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2. 客户 端 


浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辩 率 最 佳 效果 为 1024x768 像素 。 


2.3.5 文件 夹 组 织 结构 


在 进行 网 站 开发 之 前 ， 要 对 网 站 的 整体 文件 来 组 织 架构 。 | Sa 人 
进行 规划 ， 对 网 站 中 使 用 的 文件 进行 合理 的 分 类 ， 分 别 放置 “| 容 二 一 一 条 交 
于 不 同 的 文件 夹 下 。 通 过 对 文件 夹 组 织 架构 的 规划 ， 可 以 确 | 和 


月 于 存储 阿 站 前 


保 网 站 文件 目录 明确 、 条 理 清晰 ， 同 样 也 便于 网 站 后 期 的 更 | 9 一 一 一 一 一 中 约 9 六 
新 和 维护 。 本 案例 的 文件 夹 组 织 结构 如 图 2.10 所 示 。 i 


2.4 在 Linux 操作 系统 下 搭建 PHP 开发 环境 


Red Hat Linux 9 是 Linux 众多 版 本 中 比较 大 众 化 的 一 版 。 在 安装 系统 时 ， 如 果 选 择 完全 安装 或 者 选 
择 Apache、MySQL、PHP 的 安装 包 ， 则 三 者 将 被 安装 到 系统 中 ， 用 户 只 需 将 Apache 和 MySQL 服务 启 
动 就 可 以 使 用 二 者 ， 非 党 方便， 但 是 Apache 和 MySQL 的 版 本 可 能 不 是 很 理想 。 为 了 能 够 创建 一 个 良好 
的 PHP 开发 环境 ， 这 里 将 详细 介绍 自行 在 Linux 下 安装 和 配置 Apache 2+MySQL 5.0+PHP 5 的 方法 。 
首先 应 该 到 相关 官方 网 站 下 载 三 者 的 安装 包 和 libxml2 的 安装 包 : 
httpd-2.0.58.tar.gz 或 更 高 版 本 (http://httpd.apache.org/ )。 
mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 或 更 高 版 本 (http://www.mysql.com/)。 
php-5.0.0.tar.gz 或 更 高 版 本 (http://www.php.net/)。 
libxml2-2.5.10.tar.gz 或 更 高 版 本 (如 果 读 者 系统 中 libxml2 的 版 本 已 经 等 于 或 高 于 该 版 本 ， 可 
以 不 必 下 载 该 安装 包 )。 


2.4.1 Linux 下 Apache 的 安装 配置 
首先 将 下 载 的 httpd 安装 包 复 制 到 适当 的 位 置 ， 例 如 /usr/local/work 下 《〈 如 果 目 录 不 存在 ， 可 以 建 


立 该 目录 )。 打 开 Red Hat Linux 9 的 主 菜单 ， 选 择 “系统 工具 ” 在 弹出 的 菜单 中 选择 “终端 ”命令 ， 
将 打开 如 图 2.11 所 示 的 终端 对 话 框 。 


办 办 多 加 


和 说明 Linux 下 Apache、MySQL 及 PHP 的 安装 都 是 在 如 图 2.11 所 示 的 终端 命令 窗口 中 通过 命 
令 方 式 实现 的 。 


~ We x 
文 (KE) 编 久 E) ”可 看 Vj 益 病 站。 苇 玛 G) 新 8X 
troor@loca Ihos! roons 


| 
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在 该 对 话 框 中 输入 如 下 命令 进入 work 目录 : 

cd /usr/local/work 

在 work 目录 中 输入 如 下 命令 解压 httpd-2.0.58.tar.gz: 
tar xfz httpd-2.0.58.tar.gz 

进入 解压 后 的 目录 httpd-2.0.58: 

cd httpd-2.0.58 

建立 makefile， 并 将 Apache 2 安装 到 /usr/local/apache2 目录 下 : 
./configure -prefix=/usrlocal/apache -enable-module=so 
开始 编译 : 

make 

开始 安装 到 设置 的 目录 中 : 

make install 


到 此 Apache 2 的 安装 工作 完成 ， 可 以 在 每 次 启动 系统 时 通过 如 下 命令 启动 或 重新 启动 Apache 2 
服务 : 


/usr/local/apache2/bin/apachectl start 
/usr/local/apache2/bin/apachectl restart 


打开 浏览 器 ， 在 地 址 栏 中 输入 http://127.0.0.1 或 者 http://localhost， 按 Enter 键 ， 如 果 出 现 如 图 2.12 
所 示 的 页 面 ， 则 说 明 Apache 2 安装 成 功 。 


rr 已 全 半 功 ， 全 可 以 在 这 个 日 杂 中 境界 
容 ， 或 者 把 这 个 页 名 失 
这 不 是 你 想 看 见 的 页 面 吧 ? 


襄 的 设 古 ， 如 时 如何 . 外 音 询 维 
的 网 站 旷 务 加 移 件 的 开发 者 ， 不 人 


abe 六 村 已 经 包 信 在 此 妆 行 县 中 。 
熙 可以 在 便 用 pache 的 网 站 服务 名 上 ， 自 由 垃 便 用 下 面 的 图 片 。 感 到 合用 wpschel 


图 2.12 测试 Apache 服务 器 


2.4.2 Linux 下 MySQL 的 安装 配置 


将 mysql-standard-5.0.0-alpha-pc-linux-i686.targz 复制 到 /usrlocal/work 目录 下 ， 建 立 MySQL 账号 : 


groupadd mysql 


@ 
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在 组 群 中 加 入 MySQL: 

Useradd -g mysql mysql 

进入 local 目录 : 

cd /usr/local 

将 mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 解压 到 该 目录 : 
tar xfz /usr/local/work/mysql-standard-5.0.0-alpha-pc-linux-i686.tar.gz 
考虑 到 MySQL 数据 库 升 级 的 需要 ， 所 以 通常 以 链接 的 方式 建立 /usrlocalmysql 目录 : 
In -s mysql-standard-5.0.0-alpha-pc-linux-i686.targz mysql 

进入 MySQL 目录 : 

cd mysql 

在 /usr/local/mysql/data 中 建立 MySQL 的 数据 库 : 
scripts/mysql_install_db -user=mysql 

修改 文件 权限 : 


chown -R root . 
chown -R mysql data 
chgrp -R mysql . 


到 此 MySQL 安装 成 功 。 可 以 通过 在 终端 中 输入 如 下 命令 启动 MySQL 服务 : 
/usr/local/mysql/bin/mysqld_safe -user=mysql& 

启动 MySQL 后 输入 如 下 命令 查看 安装 结果 : 

/usrlocal/mysqlbin/mysql -uroot 


如 果 终 端 对 话 框 出 现 如 图 2.13 所 示 的 提示 ， 则 说 明 


MySQL 安装 成 功 。 Er 
2.4.3 Linux 下 PHP 的 安装 配置 
首先 查看 系统 中 libxml2 的 版 本 号 ， 如 果 libxml2 的 图 2.13 测试 MySQL 


版 本 号 小 于 2.5.10， 则 需要 安装 libxml2-2.5.10.tar.gz 或 更 
高 版 本 ， 因 为 PHP 5 必须 在 libxml2 的 版 本 大 于 2.5.10 的 前 提 下 才能 够 安装 。 
将 libxml2-2.5.10.tar.gz 复制 到 /usr/local/work 目录 下 ， 并 进入 该 目录 : 


cd /usr/local/work 
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解压 libxml2-2.5.10.targz: 

tar xfz libxml2-2.5.10.tar.gz 

进入 该 目录 : 

cd libxml2-2.10 

建立 makefie 并 将 libxml2 安装 到 /usr/local/libxml2 下 : 
./configure —prefix=/usr/local/libxml2 

开始 编译 : 

make 

开始 安装 到 设置 的 目录 去 : 

make install 


到 此 libxml2 安装 成 功 。 
将 php-5.0.0.tar.gz 复制 到 /usr/local/work 目录 下 ， 并 进入 该 目录 : 


cd /usr/local/work 

解压 php-5.0.0.tar.gz: 

tar xfz php-5.0.0.tar.gz 

进入 php-5.0.0 目录 : 

cd php-5.0.0 

建立 makefile: 

./configure -with-apxs2=/usrlocallapache2/bin/apxs \ 


--With-mysql=/usr/local/mysql \ 
--With-libxml-dir=/usr/local/libxml2 


开始 编译 : 


make 


开始 安装 : 


make install 
复制 php.ini-dist 或 php.ini-recommended 到 /usr/locallib 目录 ， 并 命名 为 php.ini: 


cp php.ini-dist /usr/local/lib/php.ini 


@ 
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更 改 httpd.conf 文 件 相关 设置 , 该 文件 位 于 /usr/local/apache2/conf 中 。 找 到 该 文件 中 的 如 下 指令 行 : 

Addlypeapplicationx-gzip gztgz 

在 该 指令 后 添加 如 下 指令 : 

AddType application/x-httpd-php .php .phtml 

重新 启动 Apache， 并 在 Apache 主 目 录 下 建立 文件 test.php: 

<?php 

phpinfo(); 

?> 

在 浏览 器 中 输入 http://127.0.0.1/test.php， 按 Enter 键 ， 如 果 出 现 如 图 2.14 所 示 的 页 面 ， 则 PHP 安 
装 成 功 。 


4 
说 明 Apache2 默认 主 目录 为 /usrlocal/apache2/htdocs。 


和 mT ; T 
TTS = 
图 2.14 PHP 测试 
1 区 技巧 安装 文件 的 路 径 要 遵循 一 定 的 客观 原则 ， 为 了 避免 在 Windows 和 Linux 间 移植 程序 时 带 
来 的 不 便 ， 选 择 D:\usr\local\php 的 目录 时 要 和 在 Linux 下 的 安装 目录 相 匹 配 。 建 议 最 好 不 要 选择 中 
间 有 空格 的 目录 ， 如 E:\Program Files\PHP， 这 样 做 会 导致 一 些 未 知 错误 或 崩溃 发 生 。 


2.5 ”数据库 设计 
开发 一 个 功能 完善 的 网 上 社区 离 不 开 数 据 库 的 支持 ， 只 有 拥有 了 强大 的 数据 库 ， 网 上 社区 才能 够 
存储 大 量 的 数据 信息 ， 实 现 更 多 、 更 好 的 功能 来 吸引 更 多 的 社区 成 员 。 本 节 将 对 BCTY365 网 上 社区 数 
据 库 的 设计 进行 详细 介绍 。 
2.5.1 数据 库 分 析 


BCTY365 网 上 社区 是 一 个 中 型 的 面向 软件 开发 者 的 程序 ， 考 虑 到 开发 的 成 本 、 拱 配 的 合理 性 以 及 


@ 
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操作 的 灵活 性 等 , 使 用 了 MySQL 数据 库 。 从 成 本 考虑 MySQL 数据 库 是 完全 免费 的 ， 可 以 在 网 上 免费 
下 载 ， 从 匹配 的 角度 讲 ，PHP 与 MySQL 数据 库 一 直 是 公认 的 最 佳 搭档 ，MySQL 数据 库 不 但 可 以 在 命 
令 模式 下 进行 操作 ， 而 且 还 配备 了 一 些 比较 流行 的 图 形 化 管理 工具 ， 如 phpMyAdmin 等 ， 可 以 轻松 地 
对 MySQL 数据 库 中 的 数据 进行 操作 。 


2.5.2 数据库 概念 设计 


根据 上 述 各 节 对 BCTY365 网 上 社区 系统 做 的 需求 分 析 和 系统 设计 ， 规 划 出 BCTY365 网 上 社区 的 
实体 关系 E-R 图 。 其 中 包括 注册 用 户 实体 、 发 帖 信息 实体 、 回 帖 信息 实体 、 订 单 信息 实体 和 编程 词典 
信息 实体 ， 其 他 还 有 一 些 辅助 的 实体 ， 是 对 上 述 实 体 中 内 容 的 补充 。 由 于 涉及 的 实体 较 多 ， 这 里 只 对 
注册 用 户 实体 、 发 帖 信息 实体 和 订单 信息 实体 的 E-R 图 进行 介绍 。 

1. 注册 用 户 实 体 


注册 用 户 实体 用 于 存储 用 户 注 册 信 息 ， 包 括 编号 、 用 户 名 、 真 实 姓 名 、 密 码 、 邮 箱 、 性 别 、 电 话 、 
QQ 号 码 、 家 庭 地 址 、 访 问 次 数 、 注 册 时 间 、 最 后 一 次 登录 时 间 、IP 地 址 、 邮 政 编码 、 用 户 类 型 、 密 
码 提示 问题 、 密 码 答案 、 真 实 密码 、 表 情 图 、 发 
帖 次 数 属性 。 注 册 用 户 实体 的 E-R 图 如 图 2.15 
所 示 。 


2. 发 帖 信息 实体 


发 帖 信息 实体 用 于 存储 登录 本 社区 的 会 员 
在 论坛 中 发 布 帖子 的 相关 信息 , 包括 编号 、 用 户 
名 ID、 帖 子 类 别 、 帖 子 标题 、 帖 子 内 容 、 发 帖 


时 间 、 最 后 回复 时 间 、 表 情 图 、 访 问 次 数 、 是 否 图 2.15 注册 用 户 实体 E-R 图 
顶 帖 、 上 传 图 片 属性 。 发帖 信息 实体 的 E-R 图 如 
图 2.16 所 示 。 


3. 订单 信息 实体 


订单 信息 实体 存储 用 户 在 线 购 买 时 填写 的 订单 信息 ， 包 括 编号 、 用 户 名 、 人 性别、 家 庭 地 址 、 邮 政 
编码 、QQ 号 码 、 邮 箱 、 手 机 号 码 、 电 话 号 码 、 收 货 方式 、 邮 资 、 产 品 金额 、 订 单 时 间 、 订 单 号 、 选 择 
城市 等 属性 。 订 单 信息 实体 的 E-R 图 如 图 2.17 所 示 。 


访问 次 数 


图 2.16 发 帖 信息 实体 E-R 图 图 2.17 订单 信息 实体 E-R 图 


| 
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2.5.3 创建 数据 库 及 数据 表 


在 BCTY365 网 上 社区 系统 中 应 用 的 是 db_bcty365 数据 库 ， 其 中 涉及 18 个 数据 表 ， 数 据 表 的 名 称 


和 功能 如 图 2.18 所 示 。 


本 案例 中 创建 数据 库 和 数据 表 使 用 的 是 pppMyAdmin 图 形 化 管理 工具 ， 下 面 将 介绍 数据 库 和 数据 
表 的 创建 方法 ， 以 及 在 创建 过 程 中 需要 注意 的 一 些 问题 。 


1. 数据 库 的 创建 


打开 phpMyAdmin 图 形 化 管理 工具 的 主页 ， 首 先 在 文本 框 中 输入 要 创建 的 数据 库 的 名 称 〈 如 
db_bcty365)， 然 后 在 下 拉 列 表 框 中 选择 要 使 用 的 字符 编码 格式 ， 这 里 使 用 的 是 gb2312_chinese_ci， 如 
图 2.19 所 示 。 最 后 单 击 “ 创 建 ”按钮 ， 数 据 库 创 建成 功 。 


辕 服务 器 :localhost 局 数据 库 : db_bety365 
下 类 型 村 


bb MSAM gb2312_emtn 和 生词 入 相 信 自 表 
bbhwh MBAM 中 2312_chn JR 站 
th bbs NWI9AM oh2312_chinese_cl 。 论坛 和 帖 信息 表 
th beed MMSAM 号 2312_shiese_o 。 护林 信息 夫 
hy NWSAM 9b2312_shneoe_pl 。 二 信息 表 
qm NYSAM gb2513_enese_9 。 征 见 问题 信息 表 
四 dd MNSAM gb7312_entmese_ol 。 训 必 信息 表 
reply NNISAM gb2312_cnnwse_5 。 论坛 网 由 信息 表 
th spe NMBAM p2312_ehnese_ci 于 人 升 T 才 入 自 格 
th_soft NWBAM gb2312_ehmeae_s 。 轩 促 下载 信息 表 
ttol NYSAM gb2312_enmose_s 。 社区 公 必 信息 村 
pe MISAM gb7317_enmese_p 社区 砚 类 到 信 旧 委 


ope_hg MySAM Db2312_rWnese_P 论坛 二 共 信 息 表 
也 type_small MBAM oh2312_thiese_o 。 论坛 4 信息 表 
th user MIBAM 后 2312_thinese_o 。 注 亲 用 户 信息 表 
tah NMSAM gb2312_ehinooo_ol 。 逢 包 下 载 厅 列 号 革 种 
tm_heed NWSAM gb7312_rnines9_o 。 过 怀 司 和 简介 信息 表 
jeaveword 。 NISah gb2312_rnimesa_ol 。 存 弦 客 户 反 息 


图 2.18 db_bcty365 数据 库 中 使 用 的 数据 表 
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ET 


To mv 
FLangusge 四 : 伸 坟 -chnesesmpiied 本 
phplyatmin 牢 方 站 

» {changeL ogl {CYS} lust 


phpM yAdMINn 


图 2.19 phpMyAdmin 管理 界面 


Ls 技巧 创建 数据 库 的 过 程 中 ， 尽 量 使 用 与 程序 内 容 贴 切 的 英文 字符 进行 命名 ， 有 助 于 对 数据 库 


的 理解 。 如 果 使 用 AppServ 配置 PHP 开发 环境 ， 那 么 在 创建 数据 库 时 不 需要 指定 编码 的 格式 ， 默 
认 值 为 gb2312_chinese_ci; 如 果 自 行 配置 开发 环境 ， 那 么 就 要 指定 编码 格式 为 gb2312_chinese_ci， 
否则 创建 数据 库 的 编码 格式 为 latin1_swedish_ci， 将 导致 数据 库 中 的 数据 出 现 乱 码 。 


2. 创建 数据 表 


在 成 功 创建 数据 库 以 后 ， 接 下 来 就 是 创建 数据 表 ， 这 里 以 tb_bb 编程 词典 版 本 信息 表 为 例 ， 讲 解 
如 何 创建 数据 表 ， 以 及 在 创建 数据 表 的 过 程 中 都 需要 注意 哪些 问题 。 这 里 创建 一 个 名 字 为 tb_bb 的 数 


据 表 ， 包 括 3 个 字段 ， 如 图 2.20 所 示 。 


一 局 在 数据 亩 cb_bcy365 中 创建 一 个 新 于 


输入 新 表 字段 个 数 | 


| 


十 
fT 


图 2.20 创建 tb_bb 数据 表 
ff“ 执 行 ” 按 钮 后 ， 进 入 到 如 图 2.21 所 示 的 添加 字段 信息 的 页 面 中 ， 在 此 处 对 字段 进行 详细 的 


@ 
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设置 ， 包 括 字段 名 、 数 据 类 型 、 长 度 / 值 、 属 性 、 默 认 值 、 额 外 、 主 键 和 索引 等 。 
加 服务 器 : localhost ， 昌 数据 库 : db_bctqy365 ， 国 义 :tb_bb 


i 天 型 四 长 度 放 "Nal 芋 认 | 主 
lb 


加 号 司 克 Ta 本 


poname | [rerar 本 Fa [oma T 


raeime [osewe 习 [ [mm 可 i 
指定 数据 表 的 类 型 | sogjos eine © 3 


C [wsw 了 可 > (bz chnesec 7 


2.21 添加 数据 表 中 字段 信息 


| 


[a 技巧 创建 数据 库 中 的 数据 表 时 ， 字 段 名 的 设计 尽量 要 与 数据 表 的 内 容 相 符合 ， 这 样 有 助 于 程 
序 后 期 维护 和 修改 工作 的 进行 ， 能 够 直观 地 看 出 数据 表 的 作用 。 

如 果 使 用 AppServ 配置 PHP 开发 环境 , 那么 在 创建 数据 表 时 不 需要 指定 数据 表 类 型 和 编 
码 格 式 ; 如 果 使 用 自行 配置 的 PHP 开发 环境 那么 就 要 指定 数据 表 的 类 型 为 MyISAM 和 字符 的 编 
码 格式 为 gb2312_chinese_ci， 否 则 创建 的 数据 表 类 型 为 nnoDB， 而 编码 格式 为 latin1_swedish_ci， 
将 导致 该 数据 表 中 的 数据 复制 到 其 他 机 器 上 后 不 可 用 ， 并 且 数 据 表 中 的 数据 出 现 乱 码 。 
在 创建 数据 表 的 过 程 中 ， 一 定 要 为 数据 表 指 定 一 个 主键 ， 它 是 数据 表 的 唯一 标识 。 


掌握 数据 表 的 创建 方法 后 ， 就 可 以 月 行 创建 数据 阿 天 多 车 1ocalhest ， 必 数据 库 : db_beb865 ， 加 表 :ttb_User 
表 。 由 于 本 案例 中 涉及 的 数据 表 多 达 18 个 ,这 里 不 能 。 a Re ec 人 后 ma 


对 每 个 数据 表 的 功能 设计 进行 一 一 介绍 ， 所 以 只 给 出 。 [Be een ee ee ET 
几 个 重要 的 数据 表 的 设计 效果 图 供 广大 读者 参考 ， 其 。 ep eo pop ee ee mp 
他 数据 表 请 参见 本 书 附带 的 光盘 。 数 据 表 的 设计 结构 en We oases 大 
如 图 2.22 一 图 2.24 所 示 。 aa 

(1) tb_user (注册 用 户 信息 表 ) A 有 Se 
注册 用 户 信息 表 主 要 用 于 存储 本 社区 中 会 员 的 个 。 PE 本 本 下 汪汪 本 名 ee 
人 信息 。 该 数据 表 的 结构 如 图 2.22 所 示 。 Reed i 
(2) tb_reply 论坛 回 帖 信息 表 ) 1 rt 
论坛 回帖 信息 表 主 要 用 于 存储 登录 会 员 在 本 社区 |e ewe Eomesal 轿 国 队 5 


中 回复 帖子 的 信息 。 该 数据 表 的 结构 如 图 2.23 所 示 。 
(3) tb_bccd (编程 词典 信息 表 ) 图 2.22 注册 用 户 信 息 表 结 构 
编程 词典 信息 表 主 要 用 于 存储 本 社区 的 在 线 订购 
模块 中 出 售 的 编程 词典 的 基本 信息 。 该 数据 表 的 结构 如 图 2.24 所 示 。 


园 陈 务 器 : ocalhost ;局 数据 库 : db_bety365 ， 国 表 :tb_beod 
Fa =n 可 各 所 性 mu 人 外 


国 服务 器 ; localhost ， 昼 数据 库 ; db_bcty365 ， 回 表 ;tb_reply 中 jim) 到 ulo_inerement 自动 痪 TT a 
字 话 本 a ee HW HR EE Ez Became varharco gu2312chnese_ nl 百 的 生生 
刘 bow 二 ne mer varranttg goz312.cmese nm 本 和 
wo 二 人 吕 计划 用 户 D pead errhar50 902312_chmese ol 要 Lad 
bosid it 约克 发 0 子 守 D Comtemt medumiot gv2312.chnesed 瑞 久居 简 介 
‘title varehar(200) gb2312_chinesa_ci 是 MULL 月 有 主题 Samepanl medumlat 0h2312_chmese dl 理 ea 
antant meciumied gh2312_ehress_cl 是 ML gn 2me acim 到 开间 
reatetme aelme 是 WU ste mageaddress vorhartiD go2312.chnose.o 理 人 
mark nt 是 NULL 回 各 记录 bbid in 是 MULL 全 居民 机 qom 
Photo warcharg。 gb2312_chiress 是 MULL 上 本 上 pm meal 是 ML 人 i 

图 2.23 论坛 回帖 信息 表 结 构 图 2.24 编程 词典 信息 表 结构 


@. 
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2.6 公共 模块 设计 


2.6.1 数据库 连接 文件 


在 进行 程序 开发 的 过 程 中 ， 有 很 多 地 方 都 涉及 数据 库 的 应 用 ， 在 应 用 数据 库 之 前 首先 要 与 数据 库 
建立 连接 ， 因 此 可 以 将 数据 库 的 连接 代码 作为 一 个 公共 文件 进行 存储 ， 在 需要 使 用 数据 库 连 接 文件 的 
地 方 直 接 调 用 该 文件 即 可 ， 无 须 重复 编写 相同 的 代码 ， 既 减少 了 代码 的 元 余 ， 也 便于 对 数据 库 连 接 文 
件 进行 修改 。 在 本 项 目 中 笔者 将 数据 库 的 连接 代码 存储 于 conn.php 中 。conn.php 文件 的 代码 如 下 : 

倒 程 01 ”代码 位 置 ; 光盘 \TM\02\bcty365\conn\conn.php 

<?php 

@ ”$conn=mysql_connect("localhost","root","root"); // 连 接 数 据 库 服务 器 

@ mysql_select db("db_bcty365",$conn); /| 连接 指定 的 数据 库 

© mysql_query("set names gb2312"); // 对 数据 库 中 的 编码 格式 进行 转换 ， 避 免 出 现 中 文 乱码 的 问题 


?> 


Ah 代码 贴 二 
@ mysql_conect(): 连接 MySQL 服务 器 ， 服 务 器 的 用 户 名 为 root， 密 码 为 root。 
@ mysql_select_ db0: 用 于 连接 指定 的 MySQL 数据 库 ， 数 据 库 名 为 db_bcty365。 
@ set names gb2312: 指定 数据 库 中 字符 的 编码 格式 为 gb2312。 


成 功 创 建 conn.php 文件 后 , 在 需要 进行 数据 库 操作 的 程序 中 ,就 可 以 通过 include 或 者 其 他 包含 语 
句 调用 conn.php 文件 即 可 ， 无 须 再 编写 连接 数据 库 的 程序 代码 。 应 用 include 语句 包含 conn.php 文件 
的 代码 如 下 : 
<?php 
include ("conn/conn.php"); // 包 含 数 据 库 文件 


?> 


2.6.2 将 文本 中 的 字符 转换 为 HTML 标识 符 


在 输出 数据 库 中 数据 的 过 程 中 ， 有 必要 将 数据 中 的 一 些 特殊 字符 转换 为 HTML 标识 符 ， 这 样 可 以 
避免 一 些 不 必要 的 麻烦 。 例 如 ， 在 输出 一 个 程序 的 执行 代码 的 过 程 中 ， 如 果 不 对 其 进行 转换 ， 那 么 输 
出 的 将 不 是 程序 的 代码 ， 而 是 程序 的 执行 结果 。 这 里 将 文本 中 字符 的 转换 编写 到 一 个 自 定义 函数 
unhtml0 中 ， 并 保存 到 function.php 文件 中 ， 将 其 作为 一 个 公共 模块 来 使 用 ， 当 需要 使 用 时 直接 调用 
function.php 文件 即 可 .function.php 文件 包含 两 个 自 定义 函数 , 即 unhtml0 函 数 和 msubstr0 函 数 ,unhtml0 
函数 用 于 将 数据 中 的 特殊 字符 转换 为 HTML 标识 符 ; msubstr0) 函 数 用 于 对 字符 串 进行 指定 长 度 的 截取 。 
代码 如 下 : 

倒 程 02 ”代码 位 置 ， 光盘 \TM\02\bcty365\function.php 


<?php 
function unhtml($contentX{ // 定 义 自 定义 函数 的 名 称 


® 
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@ $content=htmlspecialchars($content); /| 转换 文本 中 的 特殊 字符 

@ $content=str_replace(chr(13),"<br>",$content); /车 换 文本 中 的 换行 符 
$content=str_replace(chr(32),"&nbsp;",$content); // 蔡 换 文本 中 的 “&nbsp;” 
$content=str_replace("[L[","<",$content]; // 蔡 换文 本 中 的 大 于 号 
$content=str_replace(") )",">",$content); // 蔡 换文 本 中 的 小 于 号 
$content=str_replace("|_|"," ",$content); // 蔡 换文 本 中 的 空格 

目 retum trim($content); /删除 文本 中 首尾 的 空格 


} 

// 定 义 一 个 用 于 截取 一 段 字 符 串 的 函数 msubstr() 

function msubstr($str$start,$len){ ” //$str 指 的 是 字符 串 ，$start 指 的 是 字符 串 的 起 始 位 置 ，$len 指 的 是 长 度 

$strlen=$start+$len; /用 $strlen 存储 字符 串 的 总 长 度 〈 从 字符 串 的 起 始 位 置 到 字符 串 的 总 长 度 ) 

for($i=0;$i<$strlen,$i++)f // 通 过 for 循环 语句 ， 循 环 读 取 字 符 串 

@ if(ord(substr($str,$i,1))>0xa0X{ /如 果 字 符 串 中 首 个 字 节 的 ASCII 序数 值 大 于 0xa0， 则 表示 为 汉字 
S$tmpstr.=substr($srt,$i,2); /每 次 取出 两 位 字符 赋 给 变量 $tmpstr， 即 等 于 一 个 汉字 


SBit+; // 变 量 自 加 1 
Jelse{ // 如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 $tmpstr 
Stmpstr.=substr($str,$i,1);} 
- 
return $tmpstr; /| 输出 字符 串 
和 > 
< 人 代码 贴 二 


@ htmlspecialchars(): 将 特殊 字符 转换 成 HTML 格式 ， 而 不 会 将 所 有 字符 都 转换 成 HTML 格式 。 

@ str_ replace(mixed search,mixed replace,mixed subject,int cont): 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替 
换 ， 参 数 &count 表示 替换 字符 串 执行 的 次 数 。 

@ trim():; 删除 字符 串 中 首尾 的 空白 或 者 其 他 字符 。 

@ substr(): 从 指定 的 字符 串 中 按照 指定 的 位 置 截 取 一 定 长 度 的 字符 。 


2.7 前 台 首 页 设计 


当今 时 代 ， 很 多 人 都 十 分 重视 事物 的 第 一 印象 ， 第 一 印象 基本 上 就 决定 对 某 个 事物 的 看 法 和 态度 ， 
在 网 络 中 更 是 如 此 ， 网 站 给 人 的 第 一 印象 如 果 不 好 ， 那 么 就 会 有 很 多 人 因此 而 不 去 浏览 该 网 站 ， 无 论 
网 站 的 内 容 是 否 丰富 。 可 以 说 网 站 首页 设计 的 成 功 与 否 直 接 影响 着 整个 网 站 的 发 展 。 


2.7.1 ”前台 首页 概述 


网 站 首页 是 整个 网 站 的 脸面 ， 既 要 突出 企业 的 形象 ， 又 要 展示 出 网 站 强大 的 功能 。 如 果 网 站 首页 
设计 得 非常 成 功 ， 那 无 疑 是 为 整个 网 站 的 成 功 增添 了 一 个 夸 码 。BCTY365 网 上 社区 首页 的 设计 以 企业 
的 品牌 形象 为 基础 ， 全 力 打造 网 站 的 整体 功能 ， 重 点 推出 企业 的 软件 产品 ， 具 体内 容 如 下 。 

网 站 菜单 导航 : 包括 “首页 ””“ 技 术 支 持 ”“ 在 线 订购 ”“ 社 区 论坛 “软件 下 载 "“ 升 级 
下 载 "”“ 购 买 须知 ”“ 联 系 我 们 ”等 导航 菜单 。 

用 户 注册 和 登录 模块 : 实现 用 户 注册 、 会 员 登 录 、 找 回 密码 和 修改 密码 的 功能 。 
网 站 公告 : 主要 用 于 发 布 社区 中 的 一 些 新 消息 和 重大 事件 。 
编程 词典 模块 : 推广 企业 的 软件 产品 。 


@ 
加 办 加 


第 2 章 BCTY365 网 上 社区 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


软件 下 载 模块 : 展示 企业 提供 的 适用 版 和 免费 的 软件 产品 。 
常见 问题 模块 : 列举 出 编程 中 常见 问题 的 解决 方案 。 

社区 论坛 模块 : 浏览 社区 论坛 中 的 部 分 帖子 。 

升级 下 载 模块 : 提供 一 些 软件 的 升级 版 本 下 载 。 

上 述 内 容 就 是 BCTY365 网 上 社区 首页 中 体现 出 的 内 容 ， 为 了 更 加 直观 地 了 解 网 上 社区 首页 的 设 
计 ， 这 里 先 预览 一 下 社区 首页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 \TM\02\bcty365\index.php， 如 图 2.25 
所 示 。 

BCTY365 网 上 社区 首页 的 设计 看 上 去 很 复杂 ， 由 多 个 版 块 组 成 ， 但 实现 的 过 程 非常 简单 。 总 体 架 
构 使 用 一 个 2 行 3 列 的 表格 和 一 个 3 行 3 列 的 表格 ， 将 其 分 割 成 不 同 的 版 块 ， 然 后 使 用 脚本 语句 从 数 
据 库 中 读 取 数据 ， 最 后 将 数据 循环 输出 到 页 面 中 ， 其 中 网 站 的 头 尾 文件 使 用 include 包含 语句 调用 。 首 
页 的 框架 结构 如 图 2.26 所 示 。 


办 办 办 加 


Mee 


Munn 


EE 


图 2.25 BCTY365 网 上 社区 首页 图 2.26 网 站 首页 的 框架 结构 


2.7.2 前台 首页 技术 分 析 


作为 网 站 首页 ， 不 一 定 要 具有 什么 特殊 的 技术 或 者 功能 ， 应 该 是 以 简洁 、 鲜 明 、 突 出 企业 形象 、 
展示 网 站 的 功能 为 主 。 即 使 使 用 的 是 一 个 静态 页 面 ， 只 要 能 够 将 内 容 表 达 全 面 、 完 整 ， 那 么 这 个 首页 
设计 也 是 非常 成 功 的 。 

在 本 案例 首页 的 设计 中 ， 应 用 到 一 个 文字 循环 滚动 的 技术 ， 通 过 该 技术 来 输出 社区 中 发 布 的 公告 
信息 。 该 技术 是 通过 JavaScript 脚本 和 div 标签 来 共同 实现 的 ， 其 实现 的 原理 是 : 首先 创建 一 个 div 标 
签 ， 然 后 在 div 标签 中 输出 公告 信息 ， 最 后 通过 JavaScript 来 对 div 标签 进行 操作 ， 实 现 div 标签 中 内 
容 的 滚动 输出 。 该 技术 的 实现 在 index.php 页 中 完成 ， 其 中 使 用 的 JavaScript 脚本 的 代码 如 下 : 

©® 
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倒 性 03 ”代码 位 置 : 光盘 \TM\02\bcty365\index.php 


<script language="JavaScript"> 


marqueesHeight=222; // 定 义 输出 标签 的 高 度 
stopscroll=false; /定义 stopscroll 的 默认 值 为 false 
with(marqueesj{ /编辑 marquees 标签 的 属性 
style.width=0; /定义 初始 宽 为 0 
style.height=marqueesHeight' /定义 marquees 标签 的 高 为 222 
style.overflowX="visible"; /定义 值 为 显示 
style.overflowY="hidden"; // 定 义 值 为 隐藏 
noWrap=true; 


onmouseover=new Function("stopscroll=true"); // 当 鼠标 经 过 时 执行 stopscroll=true 
onmouseout=new Function("stopscroll=false"); // 当 鼠标 离开 时 执行 stopscroll=false 


人 
/创建 一 个 新 的 div"templayer" 与 div"marquees "进行 连接 ， 实 现 不 间断 的 循环 输出 内 容 
document.write('<div id="templayer" style="position:absolute;z-index:1;visibility:hidden"></div>"); 
preTop=0; currentTop=0; 
function init(){ 
templayer.innerHTML=""; /设置 templayer 的 初始 值 为 空 
whilettemplayeroffsetHeight<marqueesHeight){ 。“”// 判 断 当 templayer 的 高 度 小 于 marquees 的 高 度 时 
templayer.innerHTML+=marquees.innerHTML; // 将 templayer 的 值 赋 给 marquees 


} 

marquees.innerHTML=templayer.innerHTML+templayer.innerHTML;// 将 templayer 的 值 累加 

setlnterval("scrollup()",50); /| 间隔 50 毫秒 执行 一 次 scrollup() 函 数 
function scrollup()f /实现 滚动 输出 

if(stopscroll==true) return; 1/ 判断 如 果 stopscroll==true， 不 执行 循环 


preTop=marquees.scrollTop; 

marquees.scrollTop+=1; 

if(preTop==marquees.scrollTop){ 
marquees.scrollTop=templayer.offsetHeight-marqueesHeight; 
marquees.scrollTop+=1; 


人 


</script> 


在 div 标签 中 ， 主 要 是 输出 数据 库 中 存储 的 公告 信息 ， 并 且 对 输出 的 信息 进行 截取 和 替换 ， 规 范 
输出 的 内 容 。div 标签 中 的 程序 代码 如 下 : 


<div id=marquees > <!-- 创 建 一 个 div 标签 -> 
<table width="200" height="25" border="0" align="center cellpadding="0" cellspacing="0"> 
<?php // 从 数据 库 中 读 取 公 告 数据 
$sql=mysql_query("select id,title,createtime from tb_tell order by createtime desc limit 0,10",$conn); 
$info=mysql_fetch_array($sql); 


if($info==falseX{ /| 判断 当 $info==false 时 执行 下 面 的 内 容 
0 
<tr> 
<td height="25"><div align="center"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div></td> 
</tr> 
<?php J}else{ 
$i=1; // 定 义 变量 $i=1 
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dof /执行 do…while 循环 语句 

?> 

<tr> 
<td height="25" style="padding:4"> 
<a href="tellinfo.php?id=<?php echo $infolid];?>" class="a1"> 

<?php 
if($i==1){ /| 判断 当 $i==1 时 ， 将 输出 的 内 容 设置 为 红色 

echo "<font color=red>"; 
} 
echo $i.".&nbsp;"; 
echo unhtml(msubstr($info[title], 0,50)); /应 用 自 定义 函数 对 输出 的 内 容 进 行 控制 
if(strlen($infoftitle])>50)f // 当 输出 内 容 的 长 度 超过 50 个 字符 时 用 “.…” 代 替 
echo " ..."; 

} 
echo "(".str_replace("-","/",$info[createtime]).")";// 将 输出 的 公告 时 间 中 的 “-” 用 “/” 圭 代 
if($i==1){ echo "</font>"; } 

?> </a> 
</td> 

</tr> 

<?php 
9$i++; 
}while($info=mysql_fetch_array($sql)); /do…while 循环 语句 结束 

?> 

</table> 
</div> 


在 首页 中 使 用 深 动 条 是 一 个 比较 不 错 的 方法 ， 可 以 增加 网 页 的 动态 效果 ， 增 加 网 页 的 观赏 性 ， 而 
且 不 会 影响 到 网 页 的 浏览 速度 。 


2.7.3 ”前 台 首页 的 实现 过 程 


国 前台 首页 使 用 的 数据 表 : tb_tell、tb_soft、 tb_bccd、tb_bb、tb_cjwt、tb_bbs、tb_sjxz 

开发 网 站 首页 主要 就 是 连接 数据 库 ， 从 数据 库 中 读 取 数 据 ， 最 后 应 用 循环 语句 将 数据 库 中 的 数据 
输出 到 前 台 页 面 。 由 于 使 用 的 代码 较 多 ， 而 且 多 数 都 是 重复 使 用 ， 所 以 这 里 只 给 出 首页 中 公告 发 布 模 
块 的 代码 。 

公告 发 布 模块 主要 实现 从 数据 库 中 读 取 公告 数据 ， 将 数据 在 首页 中 滚动 输出 ， 并 且 对 公告 信息 的 
长 度 进行 控制 ， 保 证 内 容 的 整齐 、 规 范 。 详 细 代码 可 以 参考 本 书 光盘 中 的 TM\02\bcty365\index.php 文 
件 。index.php 文件 的 部 分 代码 如 下 : 

倒 性 04 ”代码 位 置 ， 光盘 \TM\02\bcty365\index.php 


<?php include_once("top.php"); // 获 取 头 部 文件 > 

…// 省 略 部 分 代码 

<?php 
// 读 取 数 据 库 中 公告 表 中 的 数据 
$sql=mysql_query("select id,title,createtime from tb_tell order by createtime desc limit 0,10",$conn); 
$info=mysql_fetch_array($sql); /执行 读 取 数 据 表 中 数据 的 语句 
if($info==falseX{ // 如 果 返 回 值 为 空 则 执行 下 面 的 语句 
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$5 
<tr> 

<td height="25"><div align="center"><a href="#'" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div></td> 
</tr> 


<?php 
}else{ // 如 果 返 回 值 不 为 空 ， 则 执行 下 面 的 do…while 循环 语句 
$i=1; 
dof 
?> 
<tr> 
<td height="25" style="padding:4"><a href="tellinfo.php?id=<?php echo $info[id];?>" class="a1"> 
<?php 
if($i==1X // 判 断 当 变 量 $i==1 时 ， 输 出 的 内 容 以 红色 字体 显示 
echo "<font color=red>"; 
} 


echo $i.".&nbsp;"; 
@ echounhtml(msubstr($info[title],0,50)); 
@ iffstrlen($infoltitle])>50){ // 如 果 标题 长 度 大 于 50 个 字符 ， 则 以 省 略 号 代替 
echo " ..."; 


} 
@ echo "(".str_replace("-","/",$info[createtime]).")"; // 输 出 公告 发 布 的 时 间 ， 并 且 将 其 中 的 “/” 使 用 “-” 替 换 


if($i==1X 
echo "</font>"; 
} 
?> 
</a> 
</td> 
</tr> 
<?php 
9$i++; // 变 量 自 加 1 
}while($info=mysql_fetch_array($sql)); //do…while 循环 语句 结束 
和 
i // 省 略 了 部 分 代码 
<?php 
include_once("bottom.php"); // 调 用 网 站 的 尾 文件 
?> 
4 代码 贴 十 


@ 应 用 unhtml() 和 msubstr0 〇 自 定义 函数 去 除 输 出 字符 囊 中 的 空格 和 控制 输出 字符 串 的 长 度 。 
@ strlen(): 获取 指定 字符 囊 的 长 度 。 
@ str_replace(): 实现 字符 囊 的 替换 。 


2.8 注册 模块 设计 


2.8.1 注册 模块 概述 


BCTY365 网 上 社区 系统 为 了 更 好 地 与 广大 网 民 朋 友 进 行 交 流 和 沟通 ， 创 建 了 一 个 会 员 注册 模块 。 


@. 


第 2 章 BCTY365 网 上 社区 ( Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


通过 会 员 注册 模块 ， 可 以 有 效 地 对 用 户 信息 进行 采集 ， 并 将 合法 的 用 户 信息 保存 到 指定 的 数据 表 中 ， 


实现 与 用 户 的 长 期 沟通 和 交流 。 既然 设置 了 会 员 注 
册 模 块 , 那么 在 系统 中 就 要 为 会 员 提供 一 些 特殊 的 


权限 。 在 本 系统 中 注册 会 员 可 以 拥有 如 下 权限 : 在 本 


本 社区 的 论坛 中 发 布 和 回复 帖子 、 在 技术 支持 模块 = 


过 Eap 请 了 碟 认 es 用 部。 Pi 各 永 及 红 曙 注 时 方 二 外 


ms 
中 发 表 留 言 、 在 升级 下 载 模块 中 下 载 软件 升级 包 | 
等 , 而且 可 以 进行 修改 密码 和 找 回 密码 。 会员 注 册 和 Ee 
模块 的 运行 结果 如 图 2.27 所 示 。 sf 7 
2.8.2 ”注册 模块 技术 分 析 Ta 

会 员 注册 模块 中 , 必 不 可 少 的 就 是 要 对 用 户 i 


输入 的 信息 进行 判断 , 首先 判断 用 户 填写 的 注册 信 


图 2.27 用 户 注册 模块 的 运行 结果 


息 中 哪些 是 必须 填写 的 ， 哪 些 可 以 不 填写 ， 然 后 进 


一 步 判 断 输 入 的 信息 是 否 合理 合法 ， 例 如 ， 判 断 输 入 邮编 的 格式 是 否 正确 ， 判 断 输 入 邮箱 的 格式 是 否 
正确 等 。 对 表单 中 提交 的 数据 进行 判断 最 常用 的 办 法 就 是 使 用 JavaScript 脚本 , 也 可 以 使 用 正则 表达 式 。 


下 面 讲解 在 本 模块 中 如 何 通 过 JavaScript 实现 表单 提交 数据 验证 。 


操作 原理 是 : 在 form 表单 中 调用 onsubmit 事件 ， 通 过 该 事件 调用 指定 的 JavaScript 脚本 ， 执 行 
chkinputO 自 定义 函数 ， 实 现 对 表单 中 提交 数据 的 验证 。 在 JavaScript 脚本 中 ， 实 现 对 表单 中 提交 的 数 
据 进行 判断 ， 判 断 输入 的 内 容 是 否 为 空 以 及 格式 是 否 正确 ， 如 果 正 确 则 继续 执行 ， 否 则 将 弹出 提示 对 


话 框 ， 并 将 鼠标 的 焦点 指定 到 出 错 的 位 置 。 具 体 的 JavaScript 脚本 代码 如 下 : 


<script language="JavaScript" type="text/javascript"> 


function chkinput(form){ /定义 一 个 函数 
ifform .tel.value==""){ /| 判断 tel 文本 框 中 的 值 是 否 为 空 
alert(" 请 填写 联系 电话 !"); /如 果 为 空 ， 则 输出 “请 填写 联系 电话 !” 
form.tel.select(); // 返 回 到 tel 文本 框 
return(false); 
nt // 判 断 email 文 本 框 的 值 是 否 为 空 
alert(" 请 输入 E-mail 地 址 ""); // 如 果 为 空 ， 则 输出 “请 输入 E-mail 地 址 !” 
form.email.select(); /| 返回 到 email 文本 框 
retum(false); 
} 


var i=form.email.value.indexOf("@"); 
var j=form.email.value.indexOf("."); 
// 进 一 步 判 断 邮 箱 的 格式 是 否 正确 ， 是 否 包 含 “@” 和 “.” 
if((i<O)NIHj>0)0<0) 
alert(" 请 输入 正确 的 E-mail 地 址 ""); 


form.email.select(); /| 返回 到 email 文本 框 
retum(false); 
} 
retum(true); /提交 表单 
</script> 
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上 述 代 码 中 ,只 是 列举 了 JavaScript 中 的 部 分 内 容 ， 并 且 在 对 电话 号 码 进行 判断 时 ， 只 是 判断 其 是 
否 为 室 ， 没 有 进一步 判断 电话 号 码 的 格式 是 否 正 确 。 如 果 想 要 更 加 准确 地 判断 电话 号 码 的 格式 是 否 正 
确 ， 可 以 采用 下 面 的 方法 : 通过 正则 表达 式 的 preg_match() 函 数 ， 在 表单 提交 处 理 页 中 对 电话 号 码 进行 
判断 。 

preg_match(O) 函 数 的 语法 格式 如 下 : 


int preg_match ( string pattern, string subject [, array matches [, int flags]] ) 
preg_match() 函 数 的 参数 说 明 如 表 2.3 所 示 。 


表 2.3 preg_match() 函 数 的 参数 说 明 
参 数 说 了 明 
attern 必要 参数 。 需 要 匹配 的 正则 表达 式 
必要 参数 。 输 入 的 字符 串 
可 选 参数 。 输 出 的 搜索 结果 的 数组 ， 如 Sout[0] 将 包含 与 整个 模式 匹配 的 结果 ，$out[1] 将 包含 与 第 一 个 捕 
获 的 括号 中 的 子 模式 所 匹配 的 结果 ， 依 次 类 推 
可 选 参数 。 标 记 PREG_OFFSET_CAPTURE 可 对 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 串 偏 移 量 ， 
本 标记 自 PHP 4.3.0 起 可 用 


matches 


flags 


通过 preg_match() 函 数 判 断 电话 号 码 的 格式 是 否 正确 的 方法 如 下 : 首先 定义 一 个 用 于 判断 电话 号 码 
格式 的 正则 表达 式 。 代 码 如 下 : 


人 (dt3}-)Gdt8DSIACd(4-)Gdf7DSIAGd(4-)Gdf8)9/ 


正则 表达 式 的 功能 分 析 如 下 : 使 用 “^” 和 “$” 对 字符 串 进 行 边 界 的 限制 ， 对 区 号 从 字符 串 的 开 
始 进行 匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 ; 将 括号 “0” 中 的 内 容 作为 一 个 原子 使 用 ; 使 
用 “\d” 来 匹配 一 个 数字 ， 区 号 为 3 或 4 个 数字 ， 其 他 数字 为 7 或 8 个 ; 使 用 “{}” 来 对 前 字符 进行 重 
复 匹配 ;使 用 “|” 对 匹配 的 模式 进行 选择 ， 分 成 3 个 模式 。 

然后 将 该 正则 表达 式 应 用 到 preg_match0 函 数 中 ,对 表单 提交 的 电话 号 码 进行 判断 ， 如 果 正 确 则 继 
续 执 行 ， 和 否则 弹出 提示 信息 ， 并 返回 到 表单 提交 页 。 代 码 如 下 : 


<?php 
S$tel="0431-84978981"; // 定 义 一 个 电话 号 码 的 变量 
if(preg_1 ON ehh OOO Oe A Stel,S$counts)\{ 
echo "您 输入 的 电话 号 码 格式 
jelse{ 


echo "<script>alert(' 您 输入 的 电话 号 码 的 格式 不 正确 !);history.back()</script>"; 
} 


ke 


2.8.3 ”注册 模块 的 实现 过 程 


国 ”注册 模块 使 用 的 数据 表 : tb_user 
注册 模块 的 实现 过 程 非常 简单 ， 首 先 阅读 注册 服务 条 款 ， 然 后 填写 用 户 注 册 的 用 户 名 和 密码 ， 提 
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交 后 由 系统 判断 输入 的 用 户 名 是 否 被 占用 ， 如 果 未 被 占用 则 可 以 继续 注册 ， 填 写 详细 的 注册 信息 ， 将 
数据 提交 到 表单 处 理 页 进行 处 理 ， 最 后 将 用 户 注册 的 信息 保存 到 指定 的 数据 表 中 。 用 户 注册 模块 的 实 
现 过 程 主要 由 3 个 文件 完成 : register.php 用 于 输出 注册 服务 条 款 ， 以 及 填写 注册 的 用 户 名 和 密码 ， 并 
且 判 断 注册 的 用 户 名 和 密码 是 否 被 占用 ; getuserinfo.php 文件 用 于 填写 详细 的 注册 信息 ， 并 且 在 表单 中 
应 用 数字 验证 码 技术 ; savereginfo.php 文件 用 于 对 表单 中 提交 的 数据 进行 处 理 , 将 数据 保存 到 指定 的 数 
据 表 中 。 

在 savereginfo.php 文件 中 ， 首 先 连接 数据 库 ， 然 后 获取 表单 中 提交 的 数据 ， 并 且 判断 提交 的 用 户 
名 是 否 被 占用 ， 最 后 将 提交 的 数据 进行 处 理 ， 并 将 数据 保存 到 指定 的 数据 表 中 。 程 序 代 码 如 下 : 

倒 程 05 ”代码 位 置 ， 光盘 \TM\02\bcty365\saverreginfo.php 


<?php session_start(); /初始 化 session 变量 

include_once("conn/conn.php"); // 连 接 数 据 库 

$usernc=trim($_POST[usernc]); // 获 取 注 册 的 用 户 名 

// 判 断 指定 的 用 户 名 是 否 存在 

$sql=mysql_query("select usernc from tb_user where usernc=".$usemc."",$conn); 

$info=mysql_fetch_array($sql); /l 按 指定 条 件 检索 数据 信息 

if($infol=false)f /如果 查询 结果 不 为 空 ， 则 执行 以 下 操作 
echo "<script language='javascript>alert(' 对 不 起 ， 该 昵称 已 被 其 他 用 户 使 用 !");history.back();</script>"; 
exit; 

} 

$xym=trim($_POST[xym]); /去 除 变量 两 边 的 空格 

$num=$_POSTInum]; // 接 收 变量 值 


if(strval($xym)!=strval($num)X{ 
echo "<script>alert(' 验 证 码 输 入 错误 !");window.location.href='register.php';</script>"; 


exit,} 
// 对 表单 提交 的 数据 进行 处 理 
S$truepwd=trim($_POST[pwd1]); // 获 取 真 实 密码 
$pwd=md5($truepwd); /获取 加 密 密 码 
$truename=trim($_POST[truename]); /获取 真实 姓名 
$email=trim($_POST[email]); // 获 取 邮 箱 地 址 
$sex=$_POST[sex]; // 获 取 性 别 
Stel=trim($_POSTItel]); /获取 电话 
$yb=trim($_POST[yb]): /获取 邮政 编码 
$qq=trim($_POST[qq]); /获取 QQ 
$address=trim($_POST[address]); // 获 取 地 址 
$question=trim($_POST[question]); // 获 取 提 示 问 题 
$answer=trim($_POST[answen]); /获取 问题 答案 
Sip=getenv("REMOTE_ADDR"); // 获 取 客户 端的 IP 
S$logintimes=1; // 指 定 访问 次 数 
$regtime=date("Y-m-j H:i:s"); /获取 时 间 
$lastlogintime=$regtime; 
$usertype=0; /指定 用 户 类 型 ， 默 认为 0 
$photo=$_POST["photo"]; /获取 头像 
// 将 表单 中 提交 的 数据 存储 到 数据 表 中 


if(mysql_query("insert into 
tb_user(usernc,truename,pwd,email,sex,tel,qq,address,logintimes,regtime,lastlogintime,ip,usertype,yb,question 
,answertruepwd,photo) 
values('$usernc','$truename','$pwd','$email','$sex','$tel','$qq','$address','$logintimes','$regtime','$lastlogintime',’ 


S$ip','$usertype','$yb','$question','$answer",'$truepwd','$photo')", $conn)X{ 
@ 
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session_register("unc"); 
$_SESSION["unc"]=$usemc; 
echo "<script>alert( 注 册 成 功 !);window.location.href='index.php';</script>"; 


}else{ // 如 果 添 加 操作 失败 ， 则 给 出 提示 
echo "<script language='ijavascript'>alert( 对 不 起 ,注册 失败 !");history.back();</script>"; 
exit; ll 退出 

; 

i 


2.9 技术 支持 模块 设计 


技术 支持 模块 主要 是 从 浏览 者 的 角度 进行 设计 ， 存 储 大 量 技术 问题 的 解决 方案 数据 ， 为 浏览 者 查 
阅 提供 方便 ， 而 且 设计 一 个 企业 与 客户 沟通 的 平台 ， 能 够 随时 了 解 客户 或 者 会 员 的 意见 和 需求 。 


2.9.1 技术 支持 模块 概述 


技术 支持 模块 主要 由 3 个 子 模块 组 成 ， 包 括 常见 问题 、 客 户 反馈 和 联系 方式 。 常 见 问题 模块 主要 
用 于 展示 编程 中 一 些 常 见 问题 的 解决 方案 或 者 方法 ， 为 浏览 者 解决 编程 中 的 疑难 问题 提供 方便 ;客户 
反馈 模块 主要 用 于 收集 和 获取 来 自 客户 的 需求 和 意见 ， 联 系 方式 模块 主要 用 于 展示 企业 的 形象 和 具体 
的 联系 方式 。 


2.9.2 ”技术 支持 模块 技术 分 析 


思路 是 : 从 数据 库 中 读 取 数 据 ， 获 取 数 据 总 量 ， 在 每 页 中 显示 20 条 数据 ， 根 据 数 据 总 量 和 每 页 显示 的 
条 数 对 数据 进行 分 页 处 理 ， 计 算出 有 多 少 页 和 当前 显示 的 页 码 ， 实 现 首 页 、 上 一 页 、 下 一 页 和 尾 页 之 
间 的 页 面 跳 转 。 具体 的 设计 思路 可 以 参考 cjwt.php 文件 中 的 代码 注释 和 代码 贴 士 。 cjwt.php 文件 的 程序 
代码 如 下 : 

倒 程 06 ”代码 位 置 ， 光盘 \TM\02\bcty365\cjwt.php 


<?php 
$sql=mysql_query("select count(*) as total from tb_cjwt",$conn); 。“”// 读 取 数 据 库 中 的 数据 
$info=mysql_fetch_array($sql); // 返 回 数据 


@  $total=$info[total]; 
// 判 断 字段 total 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 
if($total==0){ 


?> 
<tr> 
<td height="22" colspan="2"><div align="center"> 对 不 起 ， 暂 无 常见 问题 !</div></td> 
</tr> 
<?php 
Jelse{ 1/ 如果 不 为 空 ， 则 执行 下 面 的 内 容 
@ if(lisset($_GET["page"]) | !is_numeric($_GET["page"])\{ // 判 断 $_GET 获取 的 page 的 值 是 否 存在 
目 $page=1; // 如 果 不 存在 ， 则 设置 变量 的 值 为 1 


第 2 章 BCTY365 网 上 社区 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


Jelse{ 
@ S$page=intval($_GET["page"]); // 如 果 存 在 ， 则 获取 变量 $_GET 的 值 
} 
// 设 置 变量 $pagesize, 每 页 显示 的 数据 数据 量 为 20 
© $pagesize=20; 
if($total%$pagesize==0){ // 如 果 变 量 的 值 为 0 
@ $pagecount=intval($total/$pagesize); // 获 取 变 量 的 整数 值 
}else{ 
[2 S$pagecount=ceil($total/$pagesize); // 如 果 不 为 0， 则 获取 实际 的 整数 值 


} 
// 读 取 数 据 库 中 的 数据 ， 按 照 时 间 进 行 降 需 排列 
$sql=mysql_query("select * from tb_cjwt order by createtime desc limit ".($page-1)*$pagesize.",$pagesize 
",$conn); 
while($info=mysql_fetch_array($sql)){ 
Ts 
Wi // 省 略 部 分 代码 
<table width="600" height="25" border="0" align="center cellpadding="0" cellspacing="0"> 
<tr> 
<td width="479"><div align="left">&nbsp;&nbsp; 共 有 常见 问题 &nbsp;<?php echo $total;?>&nbsp; 条 
&nbsp; 每 页 显示 &nbsp;<?php echo $pagesize;?>&nbsp; 条 &nbsp; 第 &nbsp;<?php echo $page;?>&nbsp; 页 / 共 
&nbsp;<?php echo $pagecount;?>&nbsp; 页 </div></td> 
<td width="269"><div align="right"> 
© <a href="<?php echo $_SERVER["PHP_SELF"]?>?page=1" class="a1"> 首 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 
if($page>1) // 判 断 如 果 页 码 大 于 1 
echo $page-1; // 输 出 前 一 页 
else 
echo 1; 
?>" class="a1"> 上 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 


if($page<$pagecount) // 如 果 页 码 小 于 总 页 数 
echo $page+1; 

else 
echo $pagecount' /输出 下 一 页 


?>" class="a1"> 下 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php echo $pagecount:?>" 
class="a1"> 尾 页 </a>&nbsp;&nbsp;</div> 
</td> 
</tr> 
</table> 


4 代码 贴 十 
@ S$total: 为 数据 库 中 数据 总 的 记录 数 。 
@ isset(0 用 于 检测 变量 是 否 已 经 设置 ; is_numeric() 用 于 检测 变量 是 否 为 数字 或 者 数字 字符 囊 。 
@ $page: 变量 为 页 码 中 的 第 几 页 。 
@intval0: 获取 变量 的 整数 值 。 
@ $pagesize: 表示 在 每 页 中 显示 多 少 条 数据 。 
@ $pagecount: 表示 所 有 的 数据 可 以 分 成 多 少 页 。 
@ ceil0: 获取 变量 中 的 整数 值 ， 这 里 用 于 获取 页 码 的 整数 值 。 
@S$_SERVER["PHP SELF"]: 服务 器 变量 ， 这 里 用 于 获取 网 页 的 链接 地 址 。 
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2.9.3 ”常见 问题 的 实现 过 程 


国 常见 问题 使 用 的 数据 表 : tb_cjwt 

常见 问题 子 模 块 实现 的 主要 功能 是 展示 出 
数据 库 中 存储 的 有 关 编 程 中 遇 到 的 常见 问题 及 
解决 方案 。 其 运行 结果 如 图 2.28 所 示 。 

该 模块 由 两 个 文件 组 成 ， 一 个 是 cjwt.php 文 二 人 
件 ， 用 于 存储 创建 问题 数据 ， 详 细 内 容 可 以 参考 
2.9.2 节 ; 另 一 个 是 lookcjwt.php 文件 ， 用 于 输出 


yxoot 并 有 指定 数据库 


sjwtphp 文件 中 对 应 问题 的 详细 介绍 和 解决 广 2 
案 。 代 码 如 下 : 
倒 性 07 ”代码 位 置 ， 光盘 \TM\02\bcty365\lookcjwt.php 
<?php 
include_once("conn/conn.php"); /与 数据 库 建 立 连接 
include_once("function.php"); /调用 自 定义 函数 


// 读 取 tb_cjwt 表 中 的 数据 ， 条 件 为 jd="$_GET[id]" 
$sql=mysql_query("select * from tb_cjwt where id=".$_GET["id"]."",$conn); 
$info=mysql_fetch_array($sql); 

?> 


<table width="635" height="100" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="94" height="30"><div align="center"><strong> 问 &nbsp;&nbsp; 题 : </strong></div></td> 
<td width="541"><?php echo unhtml($info["question"]); /| 输出 问题 的 详细 内 容 ?></td> 
</tr> 
<tr> 
<td height="70"><div align="center"><strong> 解 &nbsp;&nbsp; 答 : </strong></div></td> 
<td height="70">&nbsp;<?php echo unhtml(S$info["answer"]); // 输 出 问题 的 解决 方案 ?></td> 
</tr> 
</table> 


2.9.4 客户 反馈 的 实现 过 程 


国 客户 反馈 使 用 的 数据 表 : tb_user、tb_leaveword 

客户 反馈 子 模块 为 客户 提供 了 一 个 反馈 意见 和 提出 
要 求 的 平台 ,并 且 将 提交 的 信息 存储 到 数据 库 中 。 其 运行 TA 
结果 如 图 2.29 所 示 。 = :em 


ET 


该 功能 只 对 本 网 站 中 的 会 员 开 通 , 即 只 有 以 会 员 身份 和 昌 
登录 的 用 户 才 具 有 反馈 信息 的 权限 , 其 中 在 对 提交 表单 的 。。 名 和 


细节 处 理 上 使 用 JavaScript 脚本 来 验证 表单 中 的 值 是 否 为 
空 ， 而 且 还 使 用 了 数字 验证 码 技术 。 对 表单 中 提交 的 数据 图 2.29 客户 反馈 模块 的 运行 结果 
进行 处 理 是 在 saveleaveword.php 文件 中 完成 的 , 该 文件 主 

要 用 于 对 表单 中 提交 的 数据 进行 处 理 ， 将 数据 存储 到 数据 库 中 。 代 码 如 下 : 
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倒 程 08 ”代码 位 置 : 光盘 \TM\02\bcty365\saveleaveword.php 


<?php session_start(); 1/ 初始 化 一 个 session 变量 
$xym=$_POSTIxym]; // 获 取 $_POST 提交 的 值 
if($xym!=$_SESSION["autonum"]){ // 判 断 验 证 码 是 否 正确 
echo "<script>alert(' 效 验 码 输 入 错误 !");history.back();</script>"; 
exit; 
: 
$title=$_POSTI["title"]; 1/ 获取 反馈 信息 的 标题 
$content=$_POSTI["content"]; /| 获取 反 馈 信息 的 内 容 
$type=$_POSTI["type"]; // 获 取 反 馈 信息 的 类 型 
include_once("conn/conn.php"); /与 数据 库 建 立 连接 
$sql=mysql_query("select id from tb_user where usernc=".$_SESSION["unc"]."",$conn); // 读 取 数 据 库 中 的 数据 
$info=mysql_fetch_array($sql); // 获 取 结 果 集 中 的 数组 
$userid=S$info["id"]; 
// 向 数据 库 中 添加 数据 


if(mysql_query("insert into tb_leaveword(userid,type,title,content,createtime) values('$userid''$type’,'$title’, '$content’, 
".date("Y-m-j H:i:s").")",$conn)X{ 


echo "<script>alert(' 留 言 发 表 成 功 !");history.back();</script>"; 1/ 添加 操作 成 功 ， 给 出 提示 信息 
}else{ 
echo "<script>alert(' 留 言 发 表 失 败 !");history.back();</script>"; // 添 加 操作 失败 ， 给 出 提示 信息 


} 


?> 


2.9.5 ”单元 测试 


在 对 客户 反馈 模块 进行 测试 的 过 程 中 ， 当 提交 反馈 信息 时 ， 提 示 “ 留 言 发 表 失 败 !”。 经 过 初步 的 
判断 ， 问 题 可 能 出 现在 向 数据 库 添加 数据 的 过 程 中 。 为 进 
一 步 查找 出 问题 的 根源 , 在 saveleaveword.php 页 中 应 用 了 
mysql_error() 函 数 ， 该 函数 可 获取 详细 的 错误 信息 ， 将 它 
放置 在 insert 添加 语句 之 后 ， 然 后 重新 运行 网 页 ， 运 行 结 
果 如 图 2.30 所 示 。 

在 该 结果 图 中 可 以 看 出 是 数据 表 中 的 type 字段 出 了 
问题 ， 经 过 仔细 检查 发 现 ， 原 来 是 在 编写 insert 添加 语句 
的 过 程 中 ， 将 语句 中 字段 的 排列 顺序 与 对 应 的 字段 值 的 顺序 弄 错 了 ， 如 图 2.31 所 示 。 


图 2.30 应 用 mysql_error0 函 数 获取 的 错误 信息 


图 2.31 代码 中 的 错误 截图 
调整 变量 $title 和 S$type 的 顺序 ， 重 新 运行 程序 ， 提 示 “ 留 言 发 表 成 功 !”。 


2.10 在线 订购 模块 设计 


在 线 订 购 模块 的 功能 是 实现 在 线 购买 企业 推出 的 软件 产品 ， 其 操作 的 流程 主要 通过 购物 车 和 订单 


@ 
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管理 来 实现 。 
2.10.1 在 线 订购 模块 概述 


在 线 订购 的 功能 对 所 有 访问 网 站 的 人 开放 ， 没 有 任何 的 权限 限制 。 其 操作 流程 如 图 2.32 所 示 。 


浏览 者 


图 2.32 在 线 订购 模块 的 操作 流程 
2.10.2 在 线 订购 模块 技术 分 析 


在 线 订 购 管理 模块 中 ， 不 可 缺少 的 一 项 内 容 就 是 对 订单 进行 打印 。 下 面 就 来 讲解 一 下 订单 打印 功 
能 的 实现 方法 。 在线 订购 管理 模块 中 运用 的 是 WebBrowser 打印 方法 。WebBrowser 是 IE 内 置 的 浏览 器 
控件 ， 无 须 用 户 下 载 。 其 优点 是 客户 端 独立 完成 打印 目标 文档 的 生成 ， 减 轻 服 务 器 负荷 ;缺点 是 源 文 
档 的 分 析 操 作 复 杂 ， 并 且 要 对 源 文档 中 要 打印 的 内 容 进行 约束 。 

下 面 介绍 WebBrowser 控件 的 具体 参数 ， 如 表 2.4 所 示 。 


表 2.4 WebBrowser 控件 的 具体 参数 说 明 


参数 名 称 说 了 明 
document.all.WebBrowser.Execwb(7,1): 表示 打印 预览 
document.all.WebBrowser.Execwb(6, 1): 表示 打印 
document.all.WebBrowser.Execwb(6,6): 表示 直接 打印 
document.all.WebBrowser.Execwb(8, 1): 表示 页 面 设置 
document.all.WebBrowser.Execwb(1,1): 打开 页 面 
document.all.WebBrowser.Execwb(2, 1): 关闭 所 有 打开 的 IE 窗口 
document.all.WebBrowser.Execwb(4,1): 保存 网 页 
document.all.WebBrowser.Execwb(10,1): 查看 页 面 属性 
document.all.WebBrowser.Execwb(17,1): 全 选 
document.all.WebBrowser.Execwb(22,1): 刷新 
document.all.WebBrowser.Execwb(45,1): 关闭 窗 体 无 提示 


该 技术 的 实现 原理 是 :首先 通过 onClick 事件 调用 一 个 JavaScript 脚本 ,然后 执行 openprintwindowO 


函数 ， 将 指定 的 变量 值 传递 到 订单 打印 页 中 (printwindow.php)， 最 后 在 订单 打印 页 中 实现 打印 及 打印 
预览 的 功能 。 调 用 JavaScript 脚本 和 执行 openprintwindow() 函 数 在 shopping_dd.php 页 中 完成 。 关 键 代 


@ 
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码 如 下 : 
倒 程 09 ”代码 位 置 ， 光盘 \TM\02\bcty365\shopping_dd.php 
<script language="javascript"> 
function openprintwindow(x,yzjf // 定 义 一 个 函数 ， 获 取 传 递 的 参数 
// 通 过 window 对 象 中 的 open 方法 打开 一 个 新 窗口 ， 并 设置 其 属性 
@ window.open("printwindow.php?ddno="+x+"&pv="+z,"newframe","top=200,left=200,width=635, 
height="+(230+20*y)+",menubar=no,location=no,toolbar=no, scrollbars=no,status=no"); 
} 
</script> 
<!-- -一 -一 -通过 onclick 事件 调用 JavaScript 脚本 ,传递 参数 值 。 当 参数 z 的 值 为 p 时 执行 打印 功能 一 一 -一 一- -> 
<td width="75"> <img src="images/bg_14(11).jpg" width="69" height="20" 
四 onclick="javascript'openprintwindow('<?php echo base64 decode($ GET["ddno"]);?>','<?php echo 
S$gnum;?>",'p')" style="cursor:hand"/></td> 
<!-- 一 一 -通过 onclick 事件 调用 JavaScript 脚本 ， 传 递 参 数值 。 当 参数 z 的 值 为 v 时 执行 打印 预览 功能 -一 2 
<td width="90"><img src="images/bg_14(12).jpg" width="69" height="20" onclick="javascript:openprintwindow 
(<?php echo base64_ decode($ GET["ddno"]);?>",'<?php echo $gnum;?>",'v'")" style="cursor:hand"/></td> 


Ah 代码 贴 十 
@ open(): 打开 一 个 窗口 ， 可 以 设置 其 URL、 名 称 、 大 小 、 按 钮 以 及 其 他 属性 。 该 方法 的 基本 语法 结构 如 下 : 


window.open(ur,name,features,replace) 


其 中 ，url 是 要 在 新 窗口 中 打开 文档 的 URL 地 址 ; name 是 要 打开 窗口 的 名 字 ， 用 HTML 链接 的 target 属性 进行 定 
位 时 会 有 用 ; features 是 一 个 用 过 号 分 隔 的 字符 串 ， 列 举 窗口 的 特征 ; replace 是 一 个 可 选 的 Boolean 值 ， 指 出 是 否 允 许 
URL 替换 窗口 的 内 容 ， 

@ onclick0: 表明 某 元 素 被 筷 标 单 击 触发 的 事件 。 


订单 的 打印 和 打印 预览 的 功能 在 printwindow.php 页 中 完成 ， 首 先 编写 一 个 实现 打印 预览 功能 的 
JavaScript 脚本 ， 然 后 建立 HTML 的 object 标签 ， 调 用 WebBrowser 控件 ， 最 后 获取 变量 传递 的 值 ， 当 
变量 的 值 为 p 时 执行 打印 功能 ， 当 变量 的 值 为 v 时 执行 打印 预览 的 功能 。printwindow.php 页 的 关键 代 
人 码 如 下 : 

倒 程 10 代码 位 置 ， 光盘 \TM\02\bcty365\printwindow.php 


<script> 
function printview(){ /定义 一 个 函数 
0 document.all.WebBrowser1.ExecWB(7,1) ; /| 执行 WebBrowser 控件 ， 实 现 打印 预览 
window.close(); /关闭 窗口 
} 
</script> 
<!- 一 一 -一 一 一 -一 一 一 一 一 一 一 一 一 一 建立 HTML 的 object 标签 ， 调 用 WebBrowser 控件 -一 一 一 一 一 一 一- 一 一- 一 一 > 


<object ID='WebBrowser1' WIDTH=0 HEIGHT=0 
CLASSID='CLSID:8856F961-340A-11D0-A96B-00C04FD705A2'></object> 


< 一 一 一 一 一 一 一 一 一 在 body 中 调用 onload 事件 ， 执 行 打印 操作 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 - 一 > 
四 <bodytopmargin="0" leftmargin="0" bottommargin="0" onLoad=" 
<?php 

if($_GET["pv"]=="p"X{ /判断 当 变量 值 为 p 时 ， 执 行 打印 操作 
?> 
© window.print(); 
<?php 


@ 
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}elseif($_GET["pv"]=="v")}{/ /判断 当 变 量 值 为 v 时 ， 执 行 打印 预览 操作 
Vs 

printview() 
<?php} ?> 


ed 


< 代 码 贴 十 
@ document.all.WebBrowser1.ExecWB(7,1): WebBrowser 控件 中 的 参数 ， 表 示 打 印 预 览 。 
@ onLoad: 表明 某 对 象 已 载 入 窗口 。 
四 print0: window 对 象 中 的 一 个 方法 ， 实 现 打 印 的 功能 。 


ol 
Ee 
re strta css series 下 明日 公共 从 天 上 


CT 


图 233 订单 打印 操作 的 运行 结果 
2.10.3 ”购物 车 的 实现 过 程 


国 ”购物 车 使 用 的 数据 表 : tb_bccd 
购物 车 的 功能 是 临时 储存 用 户 选 购 的 商品 ， 用 户 可 以 对 购物 车 中 的 商品 进行 添加 、 修 改 、 删 除 和 
更 新 操作 ， 也 可 以 选择 进行 结算 。 其 运行 的 结果 如 图 2.34 所 示 。 


要 


品 的 用 户 分 配 一 辆 购物 车 ， 使 其 能 够 记录 自己 已 。 | 一 Eee 
经 选 购 的 产品 。 其 工作 的 原理 与 超市 中 顾客 使 用 此 人 
购物 车 进行 购物 是 相同 的 ， 只 是 这 里 使 用 的 不 是 。 [a me nmr mar sm 


真正 意义 上 的 购物 车 而 是 两 个 session 变量 , II 


个 存储 用 户 选 购 商 品 的 ID ($goodsid)， 另 一 个 

存储 用 户 选 购 该 商品 的 数量 ($goodsnum)。 如 果 图 2.34 购物 车 的 运行 结果 图 

用 户 在 一 次 购物 中 选 购 多 种 不 同类 的 商品 ， 则 使 

用 @ 对 不 同类 商品 的 不 同 ID 和 数量 进行 分 割 。 例 如 ， 用 户 选 购 的 不 同类 商品 id 为 1、2、3， 则 session 
变量 Sgoodsid 中 存储 的 值 为 “1@2@3@”; 其 中 同一 种 商品 不 能 购买 两 次 , 如 果 想 要 购买 多 个 同 种 产品 ， 
可 以 在 购物 车 中 更 改 购买 商品 的 数量 。 在 本 案例 中 ， 购 物 车 的 分 配 功能 通过 shopping_cart_first.php 文 
件 来 完成 ， 该 文件 的 代码 如 下 : 
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倒 程 11 ”代码 位 置 光盘 \TM\02\bcty365\shopping_cart_firstphp 


@ <?php session_start(); // 初 始 化 session 变量 

@ session_register("goodsid"); // 创 建 一 个 session 变量 

©@ session_register("goodsnum"); // 创 建 一 个 session 变量 

if($_SESSION["goodsid]=="" && $_SESSION["goodsnum"]==""){ ”// 判 断 session 变量 中 的 值 是 否 为 空 
$_SESSION["goodsid"]=$_GET["id"]."@"; // 如 果 为 空 ， 则 将 商品 的 ID 赋 给 变量 
$_SESSION["goodsnum"]="1@"; // 将 商品 数量 设置 为 “1@” 

}else{ 

9 $array=explode("@",$_SESSION["goodsid"]); // 如 果 不 为 空 ， 则 使 用 @ 分 割 不 同 的 商品 ID 


// 判 断 如 果 获 取 的 ID 在 session 变量 中 已 经 存在 ， 则 提示 该 商品 已 经 被 放 入 购物 车 
© iflin array($_GET["id"],$array)X{ 
echo "<script>alert(' 该 编程 词典 已 经 被 放 入 购物 车 ! ');history.back();</script>"; 


exit; 
$_SESSION["goodsid"].=$_GET['id"]."@"; // 为 session 变量 赋值 
$_SESSION["goodsnum"].="1@"; /为 session 变量 赋值 


/将 商品 放 入 购物 车 中 ， 并 跳 转 到 购物 车 页 
echo "<script>window.location.href='shopping_cart.php';</script>"; 
?> 


Ah 代码 贴 二 

@ session_start(): 初始 化 session 变量 。 

@ session_register(): 创建 一 个 session 变量 goodsid， 存 储 商 品 ID. 

@ session_register(): 创建 一 个 session 变量 goodsnum， 存 储 购买 商品 数量 。 

@ explode(): 将 字符 囊 依 指定 的 字符 囊 或 字符 进行 分 割 。 返 回 由 字符 串 组 成 的 数组 ， 每 个 元 素 都 是 string 的 一 个 子 
串 ， 它 们 被 字符 串 separator 作为 边界 点 分 隔 出 来 。 如 果 设 置 了 limit 参数 ， 则 返回 的 数组 包含 最 多 limit 个 元 素 , 而 最 后 
那个 元 素 将 包含 string 的 剩余 部 分 ; 如 果 separator 为 空 宇 符 串 〔"" )，explode() 函 数 将 返回 false; 如 果 separator 所 包含 
的 值 在 string 中 找 不 到 ， 那 么 explode() 函 数 将 返回 包含 string 单个 元 素 的 数组 ; 如 果 参 数 limit 是 负数 ， 则 返回 除了 最 后 
的 -limit 个 元 素 外 的 所 有 元 素 。 

@ in array (): 在 指定 的 数组 中 搜索 某 个 值 ， 如 果 找 到 则 返回 true， 和 否则 返回 false。 


在 实现 购物 车 的 分 配 和 添加 商品 的 功能 后 ， 接 下 来 要 做 的 就 是 查看 购物 车 中 的 商品 ， 即 实现 购物 
车 中 商品 展示 的 功能 。 在 购物 车 的 商品 展示 中 ， 可 以 实现 很 多 的 操作 ， 如 清空 购物 车 、 删 除 购买 商品 、 
更 改 购买 商品 数量 、 继 续 购 物 和 结算 。 

购物 车 商品 展示 的 功能 主要 通过 shopping_cart.php 文件 来 完成 ， 首 先 从 session 变量 中 读 取 商 品 的 
ID 和 数量 , 然后 根据 商品 的 ID 循环 输出 购物 车 中 的 商品 , 最 后 以 商品 ZD 为 标识 符 设置 不 同 的 超 链接 ， 
执行 删除 商品 或 者 更 改 购买 商品 数量 等 操作 。 代 码 如 下 : 

倒 程 12 ”代码 位 置 ， 光盘 \TM\02\bcty365\shopping_cart.php 

<table width="720" height="47" border="0" align="center" cellpadding="0" cellspacing="1" bgcolor="#999999"> 

ee Ee // 读 取 session 变量 中 的 商品 iD， 以 @ 进 行 分 割 
$arraynum=explode("@",$_SESSION["goodsnum"]);// 读 取 session 变量 中 的 商品 数量 ， 以 @ 进 行 分 割 


$markid=0; /创建 变量 ， 初 始 值 为 0 
for($i=0;$i<count($array);$i++\{ /应 用 for 循环 语句 循环 输出 商品 ID 的 值 
if($array[$i!=""){ /| 判断 如 果 商 品 ID 的 值 不 为 空 


@ 
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$markid++; // 增 加 变量 $markid 的 值 
} 


} 
if($markid==0){ // 判 断 如 果 变 量 $markid 的 值 为 空 ， 则 输出 下 面 的 内 容 
me 
<tr> 
<td height="22" colspan="4”bgcolor="#FFFFFF"><div align="center"> 对 不 起 您 的 购物 车 中 暂 无 商品 
信息 !</div></td> 


</tr> 
<?php 
Jelse{ // 如 果 $markid 的 值 不 为 空 ， 则 执行 下 面 的 内 容 
$totalprice=0; /创建 变量 $totalprice， 初 始 值 为 0 
for($i=0;$i<count($array);$i++){ /循环 输出 数组 中 的 商品 ID 值 
if($array[$i]!="™"X 
// 根 据 获取 商品 ID 的 值 ， 从 数据 库 中 获取 对 应 产品 的 信息 
$sqlcart=mysql_query("select * from tb_bccd where id=".$array[$i]."",$conn); 
$infocart=mysql_fetch_array($sqlcart) 
?> 
<tr> 


<form name="form<?php echo $array[$i]?>" method="post" action="changegoodsnum.php"> 
<td height="22" bgcolor="#FFFFFF">&nbsp;<?php echo unhtml($infocart["bccdname"]);?></td> 
<td height="22" bgcolor="#FFFFFF"><div align="center"><?php echo number_format($infocart["price"], 
2);?> </div></td> 
<td height="22" bgcolor="#FFFFFF"><div align="center"><input type="text" name="goodsnum" value= 
"<?php echo $arraynum["$i"];?>" class="inputcss" size="8" ><input type="hidden" name="id" value="<?php 
echo $infocart["id"];?>" ></div></td> 
<td height="22" bgcolor="#FFFFFF"><div align="center"><a href="javascript:form<?php echo $array 
[$?>.submit();”class="a1"> 更 改 数量 </a>&nbsp;l&nbsp;<a href="delgoods.php?id=<?php echo sinfocart 
["id"];?>" class="a1"> 删 除 该 项 </a></div></td> 


</form> 
</tr> 
<?php 
$totalprice+=$infocart["price"]*$arraynum["$i]; 
} 
} 
} 
?> 
</table> 


2.10.4 商品 订单 的 实现 过 程 


国 商品 订单 使 用 的 数据 表 : tb_bccd、tb_ dd、tb_city 


在 确定 所 要 购买 的 商品 之 后 ， 接 下 来 要 做 的 就 是 进行 购物 结算 ， 填 写 用 户 购 物 订单 ， 将 订单 保存 


到 数据 库 中 ， 并 且 随 机 生成 一 个 订单 号 ， 作 为 订单 的 唯一 标识 。 生 成 订单 的 运行 结果 如 图 2.35 所 示 。 


订单 提交 以 后 用 户 可 以 选择 汇款 的 方式 : 一 是 选择 网 上 支付 ， 那 么 将 跳 转 到 企业 指定 的 网 上 银行 


进行 汇款 ， 汇 款 的 操作 将 在 企业 指定 的 网 上 银行 中 进行 ， 这 里 不 做 讲解 ， 二 是 选择 到 指定 的 银行 向 企 
业 提 供 的 账号 中 汇款 。 企 业 将 在 收 到 汇款 后 按照 用 户 指定 的 地 址 和 方式 将 产品 送 到 。 商 品 订单 的 生成 
和 处 理由 shopping_cart_getuserinfo.php 和 savebuyuser.php 文件 来 完成 。 


@ 
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做 当 羡 的 位 置 >> 可 时 并 于 
HH oF 


本 


EE 3 EE 


支付 的 全 各 品 计 力 


打扫 
图 2.35 ”生成 订单 的 运行 结果 
订单 处 理由 savebuyuser.php 完成 ， 首 先 连 接 数 据 库 ， 随 机 生成 一 个 订单 号 ， 然 后 获取 购物 车 中 的 
商品 信息 ， 最 后 将 商品 信息 和 订单 号 存储 到 数据 库 中 。 代 码 如 下 : 
倒 程 13 ”代码 位 置 ， 光盘 \TM\02\bcty365\savebuyuser.php 


<?php session_start(); // 初 始 化 session 变量 
include_once("conn/conn.php"); // 连 接 数 据 库 
$ddnumber=substr(date("YmdHis"),2,8).mt_rand(100000,999999); // 随 机 生成 订单 号 


$sql=mysql_query("select * from tb_city where id=".$_POST["city"]."",$conn); V// 读 取 数 据 库 中 的 城市 信息 
$info=mysql_fetch_array($sql); 
if($shfs=="1"X 1/ 判断 用 户 选择 的 送 货 方式 
$yprice=S$info[pt]; 
$shfs=" 普 通 邮递 "; 
}elseif($shfs=="2"){ 
$yprice=$info[kd]; 
$shfs=" 邮 政 特快 专递 EMS"; 


} 
$array=explode("@",$_SESSION["goodsid"]); /以 @ 来 分 割 session 变量 中 存储 的 商品 ID 
$arraynum=explode("@",$_SESSION["goodsnum"]); /以 @ 来 分 割 session 变量 中 存储 的 商品 数量 
S$totalprice=0; 
for($i=0;$i<count($array);$i++){ // 循 环 读 取 数 组 中 商品 的 1D 
if$array[$il!=")f 
$sqlcart=mysql_query("select * from tb_bccd where id=".$array[$i]."",$conn); 
$infocart=mysql_fetch_array($sqlcart); 
$totalprice+=$infocart["price"]*$arraynum["$i]; 
} 
2 
S$totalprice=$totalprice+$yprice; // 获 取 汇款 金额 
// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


if(mysql_query("insert into 
tb_dd(ddnumber,recuser,sex,address,yb,qq,email, mtel,gtel,shfs,spc, slc,yprice,totalprice,createtime ,cityid) 
values(".$ddnumber.",".$_POST["recuser"].",".$_POST["sex"].",".$_POST["address"].",".$_POST["yb"].",".$ 
_POST["qq"].",".$_POST["email"].",".$_POST["mtel"].",".$_POST["gtel"].",".$shfs.",".$_SESSION["goodsid"] 
.",".$_SESSION["goodsnum"].",".$yprice.",". $totalprice.",".date("Y-m-d H:i:s").",".$_POST["city"].")", $conn)){ 
session_unregister("goodsid"); /注销 session 变量 goodsid 
session_unregister("goodsnum"); // 注 销 session 变量 goodsnum 
echo "<script>window.location.href='shopping_dd.php?ddno=".base64_encode($ddnumber).";</script>"; 


}else{ 
@ 
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echo "<script>alert( 订 单 信息 保存 失败 ， 请 重 试 !);</script>"; 
} 
有 


2.10.5 ”单元 测试 


在 在 线 订购 模块 的 测试 过 程 中 ， 当 执行 删除 购物 车 中 某 个 商品 的 操作 时 ， 发 现 该 项 操作 没 能 正确 
执行 ， 指 定 的 商品 没有 被 删除 。 首 先 ， 从 该 功能 实现 的 思路 入 手 分 析出 错 原因 ， 并 且 仔 细 核 对 其 中 使 
用 的 变量 以 及 函数 是 否 正确 。 

删除 商品 的 操作 是 根据 商品 的 ID 来 进行 处 理 的 ， 当 单 击 “删除 ” 超 链 接 时 ， 将 指定 商品 的 ID 通 
过 变量 $ides 传 入 到 执行 删除 商品 操作 的 文件 delgoods.php 中 ， 然 后 在 该 文件 中 获取 变量 $ides 传 入 的 
ID 值 ， 并 且 根据 ID 值 完成 删除 购物 车 中 指定 商品 的 操作 。 在 查看 该 功能 实现 的 过 程 中 发 现 ， 定 义 的 
变量 $ides 与 delgoods.php 文件 中 $_GET["id"] 获 取 的 变量 值 不 一 致 ， 从 而 导致 删除 商品 的 操作 没有 正确 
执行 ， 代 码 的 出 错位 置 如 图 2.36 所 示 。 
shopping_cart. php 文 件 中 执行 删除 操作 的 内 容 


PFFPF” 


f=" javascript ba_ echo $array [和 ]?>. submit 0 ;”class="al” > 更改 数 量 </ay 
7 一 一 一 “a1" > 删除 该 项 《a> 


定义 与 获取 的 变量 值 不 一 致 | 定义 传递 ID 值 的 变量 


plode (人 Sarrayid) 
(0, Sarraynum) 


用 Joinplede ("@, 
cho “Cscript vindow, location.href=" shcpping_cart. php' ;</script>”; 
?> 


图 2.36 代码 中 的 错误 显示 


查找 出 错误 原因 ， 将 定义 的 变量 与 获取 的 变量 值 统一 使 用 id， 保 存 文件 ， 重 新 运行 程序 ， 删 除 操 
作 可 以 正常 运行 。 


2.11 社区 论坛 模块 设计 
社区 论坛 模块 为 网 站 的 浏览 者 提供 一 个 交流 的 平台 ， 以 此 来 扩大 网 站 的 影响 力 ,， 汇聚 更 多 的 人 气 ， 
宣传 企业 形象 ， 推 广 企业 产品 。 
2.11.1 ”社区 论坛 模块 概述 
社区 论坛 模块 为 浏览 者 、 会 员 、 客 户 和 企业 之 间 提 供 一 个 大 的 交流 平台 ， 根 据 身份 的 不 同 ， 分 别 
给 予 不 同 的 操作 权限 。 社 区 论坛 模块 的 操作 流程 如 图 2.37 所 示 。 


在 本 论坛 中 ， 浏 览 者 只 能 够 查看 帖子 ， 注 册 会 员 既 可 以 查看 帖子 ， 也 可 以 发 布 和 回复 帖子 ， 管 理 
员 则 具有 发 布 、 回 复 、 查 看 和 删除 的 权限 。 


@ 
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图 2.37 社区 论坛 流程 图 


2.11.2 ”社区 论坛 模块 技术 分 析 


在 社区 论坛 模块 的 实现 过 程 中 , 通过 JavaScript 脚本 和 下 拉 列 表 框 的 结合 实现 一 个 不 同 版 块 之 间 快 
速 跳 转 的 功能 ， 从 而 能 够 更 加 灵活 、 方 便 地 实现 不 同 版 块 之 间 的 跳 转 。 

下 面 分 析 该 技术 是 如 何 实现 的 。 该 技术 的 实现 综合 3 个 方面 的 内 容 ， 以 一 个 下 拉 列 表 框 为 主 ， 通 
过 PHP 语句 从 数据 库 中 读 取 数 据 作 为 下 拉 列 表 框 的 值 ， 应 用 onchange 事件 来 调用 JavaScript 脚本 ， 实 
现 不 同 版 块 之 间 的 跳 转 。 这 里 以 bbs_top.php 文件 中 的 快速 跳 转 功能 为 例 进行 分 析 。 关 键 代码 如 下 : 

倒 程 14 ”代码 位 置 ， 光盘 \TM\02\bcty365\bbs_top.php 


<!-- 创 建 一 个 下 拉 列 表 框 ， 指 定名 称 为 select_type， 并 且 设 置 其 属性 ， 通 过 onchange 事件 来 调用 JavaScript 脚 
本 文件 ， 实 现 页 面 跳 转 --> 
<select name="select_type" class="inputcss" 
©@ onChange="javascript:window.location=this.options[this.selectedIndex].value;" > 
<?php 
/通过 PHP 语句 从 数据 库 中 读 取 数据 ， 使 用 数据 的 ID 作为 下 拉 列 表 框 的 值 ,使 用 数据 的 标题 title 作为 下 拉 列 表 框 显 
示 的 内 容 
$sql=mysql_query("select * from tb_type_small order by createtime desc", $conn); 
$info=mysql_fetch_array($sql); 
if$info=="")f 
echo "<option> 暂 无 讨论 区 </option>"; 
jelse{ 
echo "<option>- 版 块 快速 跳 转 -</option>"; 
dof /应 用 do…while 循环 语句 输出 下 拉 列 表 框 中 的 值 
© echo "<option value="bbs_list.php?id=".$info[id]."">".$infol[title]."</option>"; 


} 
while($info=mysql_fetch_array($sql)); /应 用 do…while 循环 语句 结束 
} 


5 
</select> 


4 代码 贴 十 
@ onChange: 某 元 素 失去 焦点 ， 并 且 从 用 户 最 后 一 次 访问 以 来 ， 其 值 已 经 改变 。 
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location: 用 于 访问 窗口 的 当前 定位 (URL )， 既 可 被 读 取 ， 又 可 被 置换 ， 可 以 通过 其 实现 某 个 页 面 的 定位 或 者 更 新 。 
@ <option value='… >…</option>: 下 拉 列 表 框 中 输出 的 值 ， 以 及 显示 的 内 容 。 


该 技术 实现 的 运行 结果 如 图 2.38 所 示 ， 它 将 实现 从 硬件 查询 模块 跳 转 到 PHP 模块 。 


EE EE 


六 当 首位 各 


| 人 umn 


BE 


图 2.38 版 块 跳 转 功能 的 运行 结果 
2.11.3 ”论坛 分 类 的 实现 过 程 


国 ”论坛 分 类 使 用 的 数据 表 : tb_bbs、tb_type_big、tb_type_small 


论坛 分 类 可 以 分 为 两 类 : 一 是 论坛 中 大 的 版 块 分 
区 ， 分 为 综合 信息 讨论 区 、 操 作 系统 、 程 序 设计 交流 。 ge 
区 、 网 管 技术 应 用 、Web 程序 开发 和 数据 库 技术 6 个 rea mm 
版 块 , 其 数据 存储 于 tb_type_big 数据 表 中 。 二 是 对 应 cp a ane 
不 同 的 版 块 中 不 同 语言 和 技术 的 分 类 ， 分 为 11 种 ， 人 ' 。 
其 数据 存储 于 tb_type_small 表 中 。 论坛 分 类 的 运行 结 mm 
果 如 图 2.39 所 示 。 一 

论坛 分 类 的 实现 原理 很 简单 ,首先 从 tb_ type_big CEA 
表 中 读 取 6 个 版 块 中 的 数据 ， 进 行 循环 输出 ， 然 后 在 版 CC 
块 中 嵌 套 循环 ， 用 于 输出 不 用 语言 的 分 类 数据 。 该 功 CE 
能 主要 通过 bbs_index.php 文件 来 完成 ，bbs_index.php 图 2.39 论坛 分 类 的 运行 结果 
文件 的 程序 代码 如 下 : 

倒 程 15 ”代码 位 置 ， 光盘 \TM\02\bcty365\bbs_index.php 

<?php include_once("top.php"); // 调 用 网 站 头 文件 

include_once("bbs_top.php"); // 调 用 社区 论坛 的 头 文件 
?> 
<?php 


1/ 循环 输出 数据 表 tb_type_big 中 的 6 个 版 块 数据 

$sql=mysql_query("select * from tb_type_big order by createtime desc", $conn); 
$info=mysql_fetch_array($sql); 

if($info==falseX{ // 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 


Sp 省 略 了 部 分 代码 
<?php 
Jelse{ // 如 果 返 回 值 为 rue， 则 执行 do…while 循环 语句 
/* 外 部 嵌 套 循环 ， 输 出 论坛 中 的 版 块 分 类 数据 */ 
@ dof 


ee 省 略 了 部 分 代码 
<table width="750" border="0" align="center "cellpadding="0" cellspacing="1" bordercolor="#FFFFFF" 


© 
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bgcolor="#6EBEC7"> 

<?php 
/循环 输出 tb_type_small 表 中 的 不 同 语言 和 技术 的 分 类 数据 
$sql1=mysql_query("select * from tb_type_small where bigtypeid=". $info["id"]."™",$conn); 
$info1=mysql_fetch_array($sql1); 


if($info1==falseX{ /| 判断 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 
3 
so 省 略 了 部 分 代码 
<?php 
}else{ // 如 果 返 回 值 为 true， 则 执行 下 面 的 内 容 ， 输 出 该 版 块 中 对 应 语言 和 技术 的 帖子 详细 信息 
?> 
Pe 省 略 了 部 分 代码 
@ ”<?php ”上 内 部 嵌 套 循环 ， 输 出 不 同 语言 和 技术 的 分 类 */ 
dof ?> 
<tr> 
<td height="30"><font color="#666666"> 创 建 时 间 : <?php echo $info1["createtime"];?> 
</font></td> 
</tr> 
eased 省 略 了 部 分 代码 
<?php 
}while($info1=mysql_fetch_array($sql1)); 
上 内 部 嵌 套 循环 结束 */ 
二 
</table> 
<?php 
}while($info=mysql_fetch_array($sql)); 
/* 外 部 炭 套 循环 结束 ， 对 版 块 中 的 大 类 进行 输出 */ 
> 
<?php include_once("bottom.php"); 2 
< 人 代码 贴 二 


@ do…while 循环 语句 ， 对 论坛 中 大 的 版 块 分 类 进行 循环 输出 。 
@ do.…while 循环 语句 ， 对 论坛 中 一 个 版 块 的 不 同 语言 和 技术 进行 循环 输出 。 


和 5 注意 在 应 用 do…while 循环 语句 时 ，while 后 的 分 号 不 能 省 略 。 


2.11.4 “论坛 帖子 浏览 的 实现 过 程 


国 论坛 帖子 浏览 使 用 的 数据 表 : tb_ bbs、tb_user、tb reply 

论坛 帖子 浏览 主要 输出 指定 帖子 的 详细 信息 ， 包 括 发 帖 人 、 用 户 级 别 和 注册 的 时 间 ， 以 及 帖子 的 
主题 、 内 容 和 发 帖 时 间 ， 包 括 上 传 的 图 片 。 本 模块 是 用 户 权限 使 用 体现 的 最 明显 地 方 ， 可 以 分 为 3 种 
情况 : 第 一 以 浏览 者 进行 登录 ， 只 能 是 浏览 帖子 的 内 容 ， 没 有 其 他 权限 ， 第 二 以 会 员 进行 登录 ， 可 以 
对 帖子 进行 回复 ， 发 表 自 己 的 看 法 ;第 三 以 管理 的 身份 进行 登录 ， 不 但 可 以 回复 帖子 ， 而 且 可 以 对 任 
何人 发 布 和 回复 的 帖子 进行 删除 和 项 帖 的 操作 。 下 面 就 来 看 一 下 以 管理 员 身 份 进行 登录 时 都 具备 哪些 


权限 ， 运 行 结果 如 图 2.40 所 示 。 
_ 国 
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CETEEITT3 


四 200T -42-2 13.19:47 
名 相 污 下 一 下 亲本 醒 me 扣 和 二 


EE CD me 


冯 宫 请 者 DT 多 起 库 系统 开 炎 完全 手册 中 的 蕴 ! 各 


图 2.40 管理 员 浏览 帖子 的 结果 图 
论坛 帖子 浏览 的 功能 通过 bbs_looks.php 文件 完成 ， 首 先 根据 传递 的 ID 值 读 取 指定 的 帖子 数据 ， 
然后 判断 登录 用 户 的 类 型 ， 最 后 根据 用 户 不 同 的 类 型 执行 不 同 的 操作 。 代 码 如 下 : 
倒 性 16 ”代码 位 置 ， 光盘 \TM\02\bcty365\bbs_look.php 


<?php 
// 根 据 $_GET 传递 的 数据 获取 tb_bbs 中 的 数据 
$sqlb=mysql_query("select * from tb_bbs where id=".$_GET["id"]."",$conn); 
$infob=mysql_fetch_array($sqlb); 
/根据 $_GET 传递 的 数据 获取 tb_user 中 的 数据 
$sql4=mysql_query("select * from tb_user where id=".$infob["userid"]."",$conn); 
$info4=mysql_fetch_array($sql4); 
?> 
i // 省 略 了 部 分 HTML 代码 
<table width="180" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td height="22"> 用 户 级 别 : 
<?php 

// 根 据 用 户 信息 表 tb_user 中 字段 usertype 的 值 判断 该 用 户 的 类 型 

// 如 果 值 为 1 则 是 管理 员 ， 值 为 2 则 是 后 台 管 理 员 ， 值 为 0 则 是 普通 会 员 
if($info4["usertype"]=="1") echo "管理 员 ";else echo "普通 会 员 "; 


</tr> 
</table> 
i /省 略 了 部 分 HTML 代码 
<table width="500" height="200" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="35" height="20"><div align="center><img src="images/lt_15(11).jpg" width="25" height= 
"25"></div></td> 
<td><?php echo S$infob["createtime"];?></td> 
</tr> 
<tr> 
<td height="150" colspan="2"> 
<?php 
// 判 断 tb_bbs 表 中 的 字段 photo 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 
@ if($infob[photo]!=""){ 


@ 
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$photos=substr($infob[photo],2,70); /获取 图 片 在 服务 器 中 的 存储 路 径 
echo (stripslashes($infob["content"])); // 输 出 帖子 的 内 容 
echo "<img src=\"$photos\">"; /| 根据 获取 的 图 片 路 径 , 输出 服务 器 中 的 图 片 
Jelse{ 儿 I 如果 tb_bbs 表 中 的 图 片 字段 photo 为 空 ， 则 执行 下 面 的 内 容 
目 echo (stripslashes($infob["content"])); // 只 输出 帖子 的 内 容 
</td> 
</tr> 
</table> 
<table width="530" height="20" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 


<td width="239" height="30">&nbsp;</td> 
<td width="291"><img src="images/lt_15(5).jpg" width="72" height="23" style="cursor:hand" onclick=" 
<?php 
// 如 果 $_SESSION["unc"] 的 值 为 空 ， 则 不 可 以 进行 项 帖子 的 操作 
if($_SESSION["unc"]==""){ 
echo "javascript'alert( 请 先 登录 本 站 ， 然 后 进行 此 操作 ! ');window.location.href='index.php'"; 
}else{ 
/否则 将 判断 当前 用 户 的 类 型 ， 如 果 是 管理 员 则 可 以 顶 帖 
$sqlu=mysql_query("select usertype from tb_user where 
Usemc=".$_SESSION["unc"]."",$conn); 
Sinfou=mysql_fetch_array($sqlu); 


@ if($infou["usertype"]==1){ 1 如 果 用 户 的 类 型 为 1， 则 有 项 帖 的 权限 
echo "javascript:window.location.href='settop.php?id=".$infob["id"].”"; 
jelsef /1 否则 不 具备 该 权限 
echo "javascript:alert(' 对 不 起 ， 您 不 具备 该 操作 权限 ! ");"; 
} 
四 
"/>&nbsp;&nbsp; 
和 <?php 


// 判 断 当前 用 户 是 否 具有 删除 帖子 的 权限 
if($_SESSION["unc"]!="™"}{ 
/| 染 件 为 用 户 不 能 为 空 ， 并 且 是 管理 员 ， 才 具备 删除 帖子 的 权限 
$sql ysql_query("select usertype from tb_user where 
usernc="".$_SESSION["unc"] conn); 
$infou=mysql_fetch_array($sqlu); 
if($infou["usertype"]==1){ 
?> 
<img src="images/lt_15(10)jpg" onclick="javascript'if(window.confirm(' 您 确定 删除 该 帖 吗 ? 
')==true){window.location.href='bbs_delete.php?id=<?php echo $infob["id"]?>";}" style="cursor:hand"/> 
<?php 


} 
} 
?> 
</td> 
</tr> 
</table> 
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亏欠 代码 贴 二 
@ $info4["usertype"]: 判断 用 户 的 类 型 ， 如 果 值 为 1 是 管理 员 ， 否 则 为 普通 会 员 . 
四 $infob[photo]: 判断 发 布 的 帖子 中 是 否 含有 图 片 ， 如 果 有 则 输出 ， 没 有 则 不 输出 。 
@ stripslashes(): 将 应 用 addcslashes() 函 数 处 理 后 的 字符 囊 返 回 原样 。 
上 @ 判断 登录 用 户 是 否 具有 顶 帖 的 权限 。 
图 判断 登录 用 户 是 否 具有 删除 帖子 的 权限 。 


/ 
所 上 面 给 出 的 是 该 文件 的 部 分 代码 ， 主 要 讲解 了 该 功能 的 实现 方法 ， 完 整 的 代码 可 以 参考 
本 书 光盘 中 的 TM\02\bcty365\bbs_lookbbs.php 文件 。 


2.11.5 ”论坛 帖子 发 布 的 实现 过 程 


国 ”论坛 帖子 发 布 使 用 的 数据 表 : tb_bbs、tb_user 

论坛 帖子 发 布 通过 两 个 文件 来 完成 : 一 个 是 
帖子 发 布 信息 的 提交 页 bbs_pubs.php; 另 一 个 是 
对 提交 的 数据 进行 处 理 的 retrieve.php 文件 。 该 功 
能 实现 的 运行 结果 如 图 2.41 所 示 。 

在 发 布 信息 的 提交 页 中 ， 显 示 当 前 用 户 的 个 


人 信息 ， 设 置 添加 数据 表单 元 素 ， 其 中 表单 元 素 图 2.41 帖子 发 布 模块 的 运行 结果 
的 设计 如 表 2.5 所 示 。 
表 2.5 发 布 信息 页 中 使 用 的 表单 元 素 
名 称 | 元 素 类 型 重要 属性 含义 
form_bbs form method="post" action="retrieve.php" enctype="multipart/form-data”" 发 帖 表 单 
class="inputcss" style="background-color:#6EBEC7"> 
<2php 
$sql=mysql_ query("select * from tb_type_small order by createtime 
desc",$conn); 
Sinfo=mysql_fetch_array($sq]); 
这 $info 一 false){ 
echo "<option> 暂 无 讨论 区 </option>"; eS 
bbs_type select j}elsef 言 或 者 技 
dof A 
四 术 的 类 别 
<option value="<?php echo S$info[id] ;?>"<?php 这 $_GET[id] 一 Sinfo[id]) 
{echo "selected=\"selected\"";}?>><?php echo S$info[title];?></option> 
<?php 上 
while($info=mysql_fetch_array($sql)); 
}7> 
bbs title text class="inputcss" style="background-color:#6EBEC7"> 帖子 标题 
bbs head radio value="<?php echo("images/bbsface/face".($i-1).".gif"):?>" 表情 图 
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续 表 


名 称 
bbs_photo 


元 素 类 型 重要 属性 

file | id="bbs_photo" class="inputcss" style="background-color:#6EBEC7" /> 上 传 图 片 
| id="contentl" class="inputcss" style="background-color:#6EBEC7"> 帖子 内 容 
value=" 提 交 " 


contentl 
Submit 


textarea 


submit 


在 retrieve.php 页 中 对 表单 提交 的 数据 进行 处 理 ， 将 数据 存储 到 tb_bbs 表 中 ， 并 且 更 新 用 户 信息 表 
tb_user 中 pubtimes 字段 的 值 ， 其 中 还 应 用 了 图 片上 传 技术 ， 将 图 片上 传 到 服务 器 中 指定 的 文件 夹 下 。 
retrieve.php 文件 的 代码 如 下 : 

倒 程 17 ”代码 位 置 ， 光盘 \TM\02\bcty365\retrieve.php 


<?php session_start(); // 初 始 化 session 变量 
Stitle=$_POST[bbs title]; /获取 帖子 的 标题 
$content=$_POST[content1]; /获取 帖子 的 内 容 
/判断 提交 的 帖子 主题 和 帖子 内 容 是 否 为 空 */ 
if(Stitle=="){ 
echo "<script>alert( 请 输入 帖子 主题 !);history.back();</script>"; 
exit; } 
if($content=="™", 
echo "<script>alert(' 请 输入 帖子 内 容 !");history.back();</script>"; 
exit, } 
J 
include_once("conn/conn.php"); // 连 接 数据 库 
1/ 根据 $_SESSION["unc"] 的 值 读 取 数据 库 中 用 户 的 信息 
$sql=mysql_query("select * from tb_user where usemc=".$_SESSION["unc"]."",$conn); 


$info=mysql_fetch_array($sql); /检索 指定 条 件 的 数据 信息 
$userid=$infofid]; /获取 用 户 id 
qtypeid=$_POST[bbs type]; /接收 版 块 名 称 
$title=$_POST[bbs_title]; // 接 收 帖子 主题 
$content=$_POST[content1]; // 接 收 帖子 内 容 
S$head=$_POST[bbs_head]; /接收 头像 
$createtime=date("Y-m-j H:is"); // 获 取 系 统 当前 时 间 
S$lastreplytime=$createtime; // 将 当前 时 间 赋 给 变量 
$readtimes=0; 


S$link=date("YmjHis"); 
@ if($_FILESrbbs_photo]["name"]==true)j{  // 上 传 图 片 ， 判 断 文件 是 否 存 在 ， 如 果 存 在 则 执行 下 面 的 内 容 
@ S$photo_name=strtolower(stristr($_FILES["bbs_photo"]["name"],".")); /获取 图 片 后 缀 名 , 将 字符 转换 成 
小 写 
if($photo_name!=".gif" & $photo_name!=".jpg" & $photo_name!=".jpeg" ){// 判 断 图 片 的 格式 是 否 符合 要 求 
echo "<script>alert(' 您 上 传 的 图 片 格式 不 正确 !");history.back();</script>"; 


Ilse{ 
目 和 $paths1=$link.mt_rand(1000000,9999999).$photo_name; // 创 建 图 片 的 名 称 
S$photos="./upfile/". $paths1; /创建 图 片 的 存储 路 径 
@ move_uploaded _file($_FILES["bbs_photo'"]["tmp_name'"],$photos); /将 图 片 存储 到 指定 的 
文件 夹 下 
// 向 数据 库 添 加 数据 


if(mysql_query("insert into tb_bbs(userid,typeid,titlie,content,createtime,lastreplytime,head,readtimes,top, 
photo) values(".$userid.",".$typeid."," .$title.",".$content.",".$createtime.",".$lastreplytime.",".$head.", ".$readtimes.", 
'0','$photos')",$conn)}{ 
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mysql_query("update tb_user set pubtimes=pubtimes+1",$conn); /更 新 tb_user 中 pubtimes 


字段 的 值 
echo "<script>alert( 新 帖 发 表 成 功 !");history.back();</script>"; 
Jelse{ 
echo "<script>alert( 新 帖 发 表 失败 !");history.back();</script>"; 
} 
}else{ // 如 果 没 有 提交 图 片 ， 则 执行 下 面 的 内 容 


if(mysql_query("insert into tb_bbs(userid,typeid ,title,content,createtime,lastreplytime,head,readtimes,top) 
values (".$userid.",".$typeid.",".S$tile.™,".$content.”,".$createtime.",".$lastreplytime.",".$head.", ".$readtimes.",，'0')", 
$conn)){ 
mysql_query("update tb_user set pubtimes=pubtimes+1",$conn); 
echo "<script>alert(' 新 帖 发 表 成 功 !");history.back();</script>"; 
jelse{ 
echo "<script>alert(' 新 帖 发 表 失 败 !");history.back();</script>"; 
} 
» 


?> 


< 代码 贴 十 
@ $_FILES['bbs_photo']["name"]: $_FILES[] 全 局 变量 ， 获 取 表 单 提交 文件 的 原始 名 称 。 
@ strtolower(): 将 指定 的 字符 转换 为 小 写字 母 。 
stristr0: 获取 指定 字符 串 (A ) 在 另 一 个 字符 串 (B ) 中 首次 出 现 的 位 置 到 (B ) 字符 串 末 尾 的 所 有 字符 串 。 该 函 
数 如 果 执 行 成 功 则 返回 剩余 的 字符 事 ， 否 则 将 返回 false。 
@ mt_rand(): 生成 一 个 随机 数 ， 用 于 上 传 文件 的 名 称 。 
@ move_uploaded_file(): 将 指定 的 文件 上 传 到 指定 的 文件 夹 下 。 


2.11.6 ”论坛 帖子 回复 的 实现 过 程 


国 论坛 帖子 回复 使 用 的 数据 表 : tb_bbs、tb_user、tb_reply 
回复 论坛 中 的 帖子 ， 必 须 是 以 会 员 或 者 管理 员 的 身份 进行 登录 ， 和 否则 不 能 进行 帖子 的 回复 操作 ， 
其 运行 结果 如 图 2.42 所 示 。 
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2.42 ”论坛 帖子 回复 的 运行 结果 


论坛 帖子 回复 功能 的 实现 主要 通过 bbs_looks.php 和 savereply.php 两 个 文件 。 其 中 应 用 JavaScript 
脚本 对 回复 帖子 的 文本 框 进行 输出 和 隐藏 的 控制 。 在 bbs_looks.php 文件 中 ， 帖 子 回复 使 用 的 表单 元 素 


第 2 章 BCTY365 网 上 社区 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


如 表 2.6 所 示 。 
表 2.6 论坛 帖子 回复 中 的 重要 表单 元 素 
名 称 重要 属性 含义 
form_repl st" action="savereply- a i " 回复 表单 
reply title text class="inputcss" id="reply title" 可 复 帖 子 主题 
bbsid hidden value="<?php echo $infob["id"]:?>" 对 应 帖子 的 ID 
bbs head value="<?php echo("images/bbsface/face".($i-1)." .gif ):2>” 表情 图 
bbs_photo id="bbs_photo" class="inputcss" 上 传 图 片 
contentl textarea id="content1" 可 复 帖子 内 容 
Submit submit value=" 提 交 " 提交 表单 


在 帖子 回复 表单 bbs_looks.php 页 中 ， 首 先 判断 登录 用 户 是 否 具有 回复 的 权限 ， 然 后 根据 提交 的 值 


展开 回复 表单 的 文本 框 ， 在 文本 框 中 输入 回复 的 主题 和 内 容 ， 最 后 将 数据 提交 到 表单 处 理 页 
savereply.php 中 。bbs_looks.php 的 主要 代码 如 下 : 


倒 程 18 ”代码 位 置 ， 光 盘 \TM\02\bcty365\ bbs_lookbbs.php 


<script language="javascript"> 


// 设 计 回复 帖子 表格 的 输出 方式 
function show_reply(X{ /定义 一 个 函数 
if(reply_bbs1.style.display=="") /| 判断 当 display 的 值 为 空 时 
reply_bbs1.style.display="none" ; // 输 出 表格 
button_show_bbs.value=" 回 复 帖子 "; /显示 回复 帖子 
} 
else if(reply_bbs1.style.display=="none") // 判 断 当 display 的 值 为 none 时 
reply_bbs1.style.display="" ; // 不 输出 表格 
button_show_bbs.value=" 关 闭 窗口 "; // 显 示 关闭 窗口 
} 
} 
</script> 
一 一 一 一 一 一 一 一 


<img src="images/lt_15(9).jpg" width="72" height="23" id="button_show_bbs" 3 ick=" 
<?php 

if($_SESSION["unc"]=="", 

echo "javascript:alert(' 请 先 登录 本 站 ， 然 后 回复 帖子 !);window.location.href='index.php";"; 

}else{ 
?> 

show_reply() 
<?php } ?> "> 


表单 处 理 页 savereply.php 将 表单 提交 的 数据 存储 到 指定 的 数据 库 中 ， 其 实现 的 方法 与 论坛 发 布 中 


的 表单 处 理 技术 是 相同 的 ， 有 关 该 技术 的 详细 讲解 请 参考 2.11.5 节 ， 这 里 不 再 獒 述 。 


2 


1.7 ”单元 测试 
在 测试 网 上 社区 的 论坛 模块 时 ,发现 发 帖 和 回帖 时 上 传 的 图 片 不 能 够 正常 显示 , 运行 结果 如 图 2.43 


@ 


PHP 项 目 开发 全 程 实录 (第 3 版 ) 


所 示 。 分 析 错 误 原 因 ， 主 要 有 两 个 方面 : 一 是 图 片 没有 上 传 到 指定 的 服务 器 文件 夹 下 ; 二 是 上 传 成 功 
后 ， 没 能 正确 地 读 取 数 据 库 中 指定 图 片 的 路 径 。 


六 当 调匀 晤 > 社区 论 二 这 二 了 三 区 由 于 匡 EEELI2R 


2.43 ”程序 运行 错误 结果 图 


首先 ， 检 测 第 一 种 情况 ， 没 有 出 现 问 题 ， 图 片 可 以 上 传 到 指定 的 文件 夹 下 ， 并 且 图 片 的 路 径 也 可 
以 存储 到 指定 的 数据 表 中 ， 从 而 排除 了 第 一 种 情况 的 可 能 。 

然后 ,看 第 二 种 情况 ， 查 看 获取 的 图 片 路 径 是 否 正确 。 发 现在 读 取 数据 库 中 图 片 路 径 的 代码 段 中 ， 
使 用 了 错误 的 字段 名 称 ， 数据库 中 图 片 路径 存 储 使 用 的 字段 名 是 photo， 而 在 程序 代码 段 中 使 用 的 却 是 
photos。 错 误 代码 如 下 : 


<?php 
if$infob[photo]!="")f /判断 是 否 存 在 图 片 
$photos=substr($infob[photos],2,70); // 获 取 图 片 存储 的 路 径 
echo (stripslashes($infob["content"])); /输出 图 片 的 内 容 
echo "<img src=\"$photos\">"; /| 输 出 图 片 
jelse{ 
echo (stripslashes($infob["content"])); // 输 出 帖子 内 容 
?> 


将 代码 段 中 的 字段 名 进行 修改 ， 然 后 重新 运行 程序 ， 图 片 正常 显示 。 
2.12 后 台 首 页 设计 


作为 一 个 完整 的 网 上 社区 系统 ， 要 想 能 够 及 时 地 对 网 站 进行 管理 和 维护 ， 必 须 具 有 一 个 强大 的 后 
台 管 理 系统 ， 对 网 上 社区 系统 中 的 数据 进行 更 新 和 维护 。 


2.12.1 后 台 首 页 概述 


网 上 社区 系统 的 后 台 管 理 采用 的 是 一 种 简单 的 框架 结构 ， 通 过 switch 语句 来 实现 。 其 包括 如 下 有 具 
软件 试用 管理 : 包括 软件 试用 产品 的 添加 和 删除 。 
编程 词典 管理 : 包括 编程 词典 版 本 的 添加 、 删 除 和 编程 词典 内 容 的 添加 和 删除 。 
在 线 订 购 管理 : 主要 用 于 管理 用 户 提交 的 订单 。 
软件 升级 管理 : 包括 升级 包 的 添加 、 删 除 和 序列 号 的 添加 和 删除 。 


' 
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回 ”站 内 公告 管理 : 主要 用 于 添加 和 删除 站 内 公告 。 
技术 支持 管理 : 主要 用 于 添加 常见 问题 和 删除 常见 问题 ， 以 及 对 客户 反馈 信息 进行 管理 。 


本 案例 中 提供 的 后 台 首页 如 图 2.44 所 示 。 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\02\bcty365\admim\ 
index.php。 


生路 oceanesms coz en eT €) KHER 


图 2.44 BCTY365 网 上 社区 系统 后 台 首 页 
2.12.2 后台 首 页 技术 分 析 


网 上 社区 后 台 首 页 的 设计 主要 应 用 switch 语句 和 include 包含 语句 ， 其 实现 的 原理 是 : 应 用 switch 
语句 ， 根 据 超 链接 中 传递 的 变量 值 进行 判断 ， 根 据 不 同 的 变量 值 应 用 include 调用 不 同 的 子 文件 。 该 技 
术 的 实现 流程 如 图 2.45 所 示 。 


《一 全 | 应用 inchde 根 据 不 同 的 
的 


图 2.45 网 上 社区 后 台 首 页 设计 流程 
为 了 能 够 更 好 地 理解 这 个 技术 ， 先 来 了 解 一 下 switch 语句 。 该 语句 的 格式 如 下 : 


switch( expr ){ llexpr 条 件 为 变量 名 称 
case expr1: licase 后 的 expr1 为 变量 的 值 
statement1; // 冒 号 (:) 后 的 是 符合 该 条 件 时 要 执行 的 部 分 
break ; /应 用 break 来 跳 离 循环 体 
case expr2 : 
statement2 ; 
break ; 
default: 
statementN; 


break; 
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参数 expr 是 表达 式 的 值 ， 即 switch 语句 的 条 件 变 量 的 名 称 ， 参数 exprl 放置 于 case 语句 之 后 ， 是 
要 与 条 件 变量 expr 进行 匹配 的 值 中 的 一 个 ;statementl 是 在 参数 exprl 的 值 与 条 件 变 量 expr 的 值 相 匹 
配 时 执行 的 代码 ; break 语句 实现 终止 语句 的 执行 ， 即 当 语 句 在 执行 过 程 中 ， 遇 到 break 就 停止 执行 ， 
跳出 循环 体 ，default 是 case 的 一 个 特例 ， 匹 配 了 任何 其 他 case 都 不 匹配 的 情况 ， 并 且 是 最 后 一 条 case 
语句 。 

通过 switch 和 include 语句 来 实现 后 台 管 理 功能 的 设计 是 一 个 很 好 的 方法 ,不 但 实现 过 程 简单 ， 而 
且 操 作 也 非常 灵活 。 关 键 代码 如 下 : 

倒 程 19 ”代码 位 置 ， 光 盘 \TM\02\bcty365\ admin\Wwzdh.php 


<?php 
Switch($htgl){ /根据 变 量 提交 的 不 同 值 
case "添加 编程 词典 版 本 ": // 判 断 与 变量 提交 的 值 是 否 相同 
include("addbb.php"); 1/ 如果 值 相同 ， 则 调用 指定 的 文件 
break; // 并 且 跳出 本 次 循环 
case "编辑 编程 词典 版 本 " 
include("editbd.php"); 
break; 
罗 // 部 分 代码 省 略 
case ™: // 当 变量 的 值 为 空 时 
include("edittell.php"); // 调 用 该 文件 
break; 
} 
?> 


2.12.3 ”后 台 首 页 的 实现 过 程 


国 ”后 台 首 页 使 用 的 数据 表 : tb_bb 

在 后 台 首 页 的 设计 过 程 中 ， 以 switch 循环 语句 为 基础 ， 架 设 整个 后 台 管 理 功 能 的 框架 结构 ， 充 分 
发 挥 include 包含 语句 的 作用 ， 调 用 不 同 的 文件 执行 不 同 的 管理 操作 ;应 用 JavaScript 脚本 来 控制 栏目 
列表 的 输出 和 隐藏 。 

控制 栏目 列表 的 输出 和 隐藏 在 menu.php 文件 中 进行 ， 首 先 定义 一 个 函数 change() 用 于 控制 表格 的 
输出 和 隐藏 ， 然 后 在 表格 中 应 用 onclick 事件 传递 不 同 的 值 到 自 定义 函数 change()， 最 后 根据 不 同 的 值 
显示 不 同 的 内 容 。 关 键 代码 如 下 : 

倒 程 20 ”代码 位 置 : 光盘 \TM\02\bcty365\admin\menu.php 


<script language="javascript"> 


// 通 过 脚本 语言 控制 文本 框 的 伸展 和 收缩 


function change(x,yX{ /定义 一 个 函数 

说 x.style.display=="none")f // 判 断 当 样式 的 值 为 none 时 
x.style.display="; // 输 出 样式 的 值 为 空 

else if(x.style.display==""){ // 判 断 当 样式 的 值 为 空 时 
x.style.display="none"; // 输 出 样式 的 值 为 none 
y.background="images/bg_16_11.jpg"; /| 输出 背景 图 片 

} 
} 


& 
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</script> 
<table width="175" height="28" border="0" align="center cellpadding="0" cellspacing="4" 
onclick="change(tz1,img_tz1)" style="cursor:hand"> 

<tr> 

<td background="images/bg_16_11.jpg" id="img_tz1" class="a4"><div align="left"><img 

src="images/bg_16_21.jpg">&nbsp; 编 程 词 典 管理 </div></td> 

</tr> 
</table> 
<table name="tz1" id="tz1" width="170" height="40" border="0" align="center" cellpadding="0" cellspacing="0" 
<?php 
// 根 据 变量 的 值 选择 执行 的 内 容 ， 当 变量 的 值 不 为 真 时 ， 隐 藏 该 表格 
if(!($_GET[htgl]j==" 添 加 编程 词典 版 本 " ||$_GET[htgl]==" 编 辑 编程 词典 版 本 " 上 |$_GET[htgl]==" 添 加 编程 词典 " 
|1$_GET[htgI]==" 编 辑 编程 词典 " )){ 


?> 
style="display:none”" 
<?php } ?> 
> 
<tr> 


<td width="40" height="24" background="images/bg_16_16.jpg">&nbsp;</td> 
<td width="114" background="images/bg_16_16.jpg"><div align="left"><a href="default.php?htgl= 添 加 编程 
词典 版 本 "> 添加 编程 词典 版 本 </a></div></td> 
</tr> 
</table> 


/ 
ww 
bcty365\admin\ 文 件 夹 下 的 相关 文件 。 


2.13 ”编程 词典 管理 模块 设计 


本 模块 的 功能 是 对 网 站 中 的 编程 词典 进行 管理 ， 包 括 添加 编程 词典 版 本 、 编 辑 编程 词典 版 本 、 添 
加 编程 词典 和 编辑 编程 词典 。 


2.13.1 编程 词典 管理 模块 概述 


本 模块 的 主要 功能 是 管理 网 站 中 在 线 出 售 的 编程 词典 软件 ， 实 现 对 编程 词典 软件 的 及 时 更 新 和 维 
护 ， 其 管理 的 内 容 主要 包括 添加 和 编辑 编程 词典 的 版 本 ， 添 加 和 编辑 编程 词典 的 详细 信息 。 在 添加 编 
程 词典 时 ， 包 括 名 称 、 版 权 、 图 片 、 类 别 、 内 容 简 介 和 不 同 版 本 的 共同 点 ; 编辑 编程 词典 包括 版 本 、 
价格 、 简 介 、 功 能 和 服务 ， 其 中 每 一 个 编程 词典 软件 只 可 以 编辑 一 次 ， 不 可 以 进行 重复 编辑 ， 如 果 要 
重新 编辑 ， 就 必须 将 已 经 编辑 过 的 信息 删除 。 


2.13.2 ”编程 词典 管理 模块 技术 分 析 


在 编程 词典 管理 模块 中 ， 应 用 到 图 片上 传 技术 ， 通 过 该 技术 将 编程 词典 的 界面 效果 上 传 到 服务 器 


_ 国 
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的 指定 文件 夹 下 。 该 技术 主要 通过 move_uploaded_file() 函 数 来 实现 ， 其 中 还 应 用 到 is_dir()、mkdir() 函 
数 ， 判 断 指定 的 文件 夹 是 否 存在 和 创建 文件 夹 ， 还 有 mt_rand0、strstr0 函 数 和 $_FILES[] 全 局 变量 。 为 
了 更 好 地 理解 和 掌握 图 片上 传 处 理 技术 ， 这 里 以 编程 词典 模块 中 的 savebccd.php 文件 为 例 进行 讲解 。 
首先 应 用 is_dir0 函 数 判断 在 服务 器 中 是 否 存在 指定 的 文件 夹 ， 如 果 不 存在 ， 则 应 用 mkdirO 函 数 创 
建 一 个 新 的 文件 夹 。 
然后 应 用 $_FILES[] 全 局 变量 获取 图 片 名 ， 应 用 strstr0) 函 数 获 取 图 片 文件 的 后 缀 名， 为 了 避免 出 现 
同名 文件 覆盖 ， 这 里 应 用 系统 的 当前 时 间 和 mtrand() 函 数 获取 的 一 个 7 位 随机 数字 作为 图 片 的 名 称 。 
最 后 确定 图 片 在 服务 器 中 存储 的 路 径 ， 将 图 片上 传 到 指定 的 文件 夹 下 。 而 数据 库 中 存储 的 数据 是 
图 片 在 服务 器 中 的 路 径 ， 当 需要 输出 图 片 时 ， 只 需要 获取 到 数据 库 中 图 片 的 路 径 即 可 。savebccd.php 
文件 的 代码 如 下 : 
倒 避 21 ”代码 位 置 ， 光 盘 \TM\02\bcty365\admin\savebccd.php 
<?php include_once("../conn/conn.php"); /| 连接 数据 库 
$bccdname=$_POST[bccdname]; // 获 取 POST 方法 提交 的 值 
$owner=$_POST[owner]; 
$typeid=$_POST[typeid]; 
$content=$_POST[content]; 
$samepart=$_POST[samepanm]; 


$addtime=date("Y-m-j H:i:s"); // 获 取 当 前 时 间 

@ ifis_dir("./bccdimages")==false)}{ // 判 断 指定 的 文件 是 否 存 在 

@ mkdir("/bccdimages"); // 如 果 不 存在 ， 则 创建 一 个 新 的 文件 夹 
} 

$link=date("YmjHis"); // 获 取 当 前 时 间 


// 为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 

// 其 中 使 用 $_FILES 获取 表单 中 真实 的 名 称 ， 使 用 strstr() 函 数 获取 文件 的 后 缀 

©@ $path=$link.mt_rand(1000000,9999999).strstr($_FILES["imageaddress"]["name"],"."); 
$address="./bccdimages/".$path; /定义 文件 上 传 的 路 径 

@ move_uploaded_file($_FILES["imageaddress"]["tmp_name"],$address);”// 将 文件 上 传 到 指定 的 文件 
中 


S$imageaddress="./admin/bccdimages/".$path; // 获 取 上 传 文件 在 服务 器 中 的 存储 路 径 
// 将 表单 中 提交 的 数据 存储 到 数据 库 中 
$query=mysql_query("insert into tb_bccd(bccdname,owner,typeid,content,samepart,imageaddress,addtime) 
values('$bccdname','$owner','$typeid','$content','$samepart','$imageaddress','$addtime')",$conn); 
if($query==trueX{ 

echo "<script>alert( 编程 词典 添加 成 功 ! ');history.back();</script>"; 


}else{ 
echo "<script>alert( 编 程 词典 添加 失败 ! ');history.back();</script>"; 
} 


> 


< 人) 代码 贴 十 
@is dir0: 判断 指定 的 文件 夹 是 否 存 在 ， 如 果 存 在 则 返回 true， 否 则 返回 false。 
@ mkdir(): 创建 一 个 新 的 文件 夹 。 
上 @ mt_rand(): 根据 提供 的 参数 min 和 max 生成 随机 数 ,如 果 没有 提供 可 选 参 数 min 和 max, 则 返回 0 到 RAND MAX 
之 间 的 伪 随机 数 。 
strstr(): 获取 一 个 指定 字符 囊 在 另 一 个 字符 囊 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字符 事 。 如 果 执 行 成 功 ， 则 返回 
剩余 字符 囊 ( 存在 相 匹 配 的 字符 ); 如 果 没 有 找到 相 匹 配 的 字符 ， 则 返回 false。 


@ 
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$_FILES[]: 全 局 变量 ,获取 所 有 上 传 文件 的 信息 .该 全 局 变量 还 可 以 获取 到 其 他 的 值 ,其 中 $_FILES[imageaddress'] 
[mame'] 获 取 的 是 客户 端 机 器 文件 的 原名 称 ; $_ FILES['imageaddress'][size] 获 取 已 上 传 文件 的 大 小 , 单位 为 字 节 ; $_FILES 
[imageaddress']['tmp_name'] 获 取 文 件 被 上 传 后 在 服务 端 存储 的 临时 文件 名 ; $_FILES[imageaddress'"]['error] 获 取 和 该 文件 
上 传 相关 的 错误 代码 。 
@ move_ uploaded file(string filename,string destination): 应 用 POST 方法 实现 文件 的 上 传 ， 参 数 filename 指定 要 上 
传 的 文件 地 址 ; 参数 destination 指定 文件 上 传 到 服务 器 后 的 存储 目录 及 名 称 。 


po 


"multipart/ form-data"” 。 


2.13.3 ”添加 编程 词典 的 实现 过 程 
国 ”添加 编程 词典 使 用 的 数据 表 : tb_bccd、tb_type 


添加 编程 词典 的 功能 是 向 数据 库 中 添加 编程 词典 的 详细 信息 ， 包 括 编程 词典 的 名 称 、 版 权 、 图 片 、 
类 别 、 内 容 简介 和 不 同 版 本 的 共同 点 。 其 运行 结果 如 图 2.46 所 示 。 


编程 记 典 名 聊 ; 
版 可 所 有 : 


FiF 纺 程 词 奥 
十条 有 明日 科技 有 限 从 本 


EE 


Er 

EE 
A 的 技术 ,为 大 的 PIF 爱好 圭 提 供 一 司 | 
Ga 


加 
[RE 不 同和 版 本 , 黄玉 及 到 不 同 直 划 程度 的 知识 司 


不 网 本 相同 点 : 


加 


添加 | 重 写 | 
图 2.46 添加 编程 词典 模块 的 运行 结果 


添加 编程 词典 信息 模块 主要 通过 addbccd.php 和 savebccd.php 文件 来 完成 ， 其 中 在 addbccd.php 文 
件 中 主要 是 设计 表单 元 素 ， 而 savebccd.php 文件 主要 是 对 表单 中 提交 的 数据 进行 处 理 。addbccd.php 文 
件 中 使 用 的 表单 元 素 如 表 2.7 所 示 。 
表 2.7 添加 编程 词典 页 中 使 用 的 重要 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 

ior Se motiods po action="savebccd.php" onSubmit="retum chkinput(this)" 编程 词典 表单 
enctype="multipart/form-data"> 

bccdname text Size="25" class="txt grey" 编程 词典 名 称 

Owner text Size="25" class="txt grey" 版 权 所 有 者 

imageaddress file Size="25" class="txt grey" 界面 图 片 
<?php include_once("../conn/conn.php"); 

S$sql=mysql_query("select * from tb type order by createtime desc",$conn); | 选择 编程 词典 

typeid 的 S$info=mysql_fetch_array($sql); 的 版 本 

if($info==false){ echo "<option > 暂 无 类 别 </option>"; 
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}else{ 


dof 选择 编程 词典 的 
0 et echo "<option value=".$info[id].">".$info[typename]."</option>"; } 版 本 
while($info=mysql_fetch_array($sqD));} ?> 
content | textarea rows="10" cols="65" class="textarea" 内 容 简 介 
samepart | textarea rows="10" cols="65" class="textarea" 不 同 版 本 的 特点 
Submit submit value=" 添 加 " class="btn_grey" 提交 表单 


savebccd.php 文件 实现 对 表单 中 提交 的 数据 进行 处 理 ， 首 先 通过 $_POST 获取 表单 中 提交 的 数据 ， 
然后 判断 指定 的 文件 夹 是 否 存 在 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 关 键 代码 如 下 : 

倒 程 22 ”代码 位 置 ， 光盘 \TM\02\bcty365\admin\savebccd.php 

<?php include_once("../conn/conn.php"); /| 连接 数据 库 

$bccdname=$_POST[bccdname]; /获取 POST 方法 提交 的 值 

$owner=$_POST[owner]; 

$typeid=$_POST[typeid]; 

$content=$_POST[content]; 

$samepart=$_POST[samepat]; 


$addtime=date("Y-m-j Hii:s"); // 获 取 当 前 时 间 

@ ifis_dir("./bccdimages")==false}{ /| 判断 指 定 的 文件 是 否 存 在 

@ mkdir("/bccdimages"); /| 如果 不 存在 ， 则 创建 一 个 新 的 文件 夹 

} 

S$link=date("YmjHis"); /| 获取 当 前 时 间 

// 为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 ， 其 中 使 用 $_FILES 获取 表单 中 真实 的 名 称 ， 使 用 
strstr() 函 数 获 取 文 件 的 后 缀 

©®@ $path=$link.mt_rand(1000000,9999999).strstr($_FILES["imageaddress"]["name"],"."); 
$address="./bccdimages/".$path; // 定 义 文件 上 传 的 路 径 
move_uploaded_file($_FILES["imageaddress"]["tmp_name"],$address);”// 将 文件 上 传 到 指定 的 文件 中 
S$imageaddress="./admin/bccdimages/".$path; /获取 上 传 文件 在 服务 器 中 的 存储 路 径 

// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


$query=mysql_query("insert into tb_bccd(bccdname,ownertypeid,contentsamepart,imageaddress,addtime) 
values('$bccdname','$owner','$typeid','$content','$samepart','$imageaddress','$addtime')",$conn); 
if($query==trueX{ 

echo "<script>alert( 编 程 词典 添加 成 功 !");history.back();</script>"; 


}else{ 
echo "<script>alert( 编 程 词典 添加 失败 !");history.back();</script>"; 
. 


Ty 


多 代码 贴 十 
@is dir0: 判断 指定 的 文件 是 否 存 在 。 
@ mkdir(): 创建 一 个 新 的 文件 夹 。 
@ mt_rand(): 获取 随机 数字 。 
strstr(): 获取 一 个 指定 字符 囊 在 另 一 个 字符 囊 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字符 事 。 如 果 执 行 成 功 ， 则 返 
回 剩余 字符 事 (存在 相 匹 配 的 字符 ); 如 果 没 有 找到 相 匹配 的 字符 ， 则 返回 false。 


@© 
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2.13.4 ”编辑 编程 词典 的 实现 过 程 


国 编辑 编程 词典 使 用 的 数据 表 : tb bccd、tb bb、tb_bbqb 
在 完成 对 编程 词典 信息 的 添加 后 ， 接 下 来 就 可 以 对 编程 词典 的 版 本 信息 进行 编辑 ， 主 要 添加 版 本 
信息 、 价 格 、 简 介 、 功 能 和 推出 的 服务 。 该 模块 的 运行 结果 如 图 2.47 所 示 。 
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图 2.47 编辑 编程 词典 模块 的 运行 结果 


该 功能 的 实现 同样 通过 两 个 文件 : 一 个 是 提交 表单 的 文件 editbccd.php; 另 一 个 是 处 理 表单 提 交 数 
据 的 文件 sacvbccdbb.php。 提 交 表 单 文件 editbccd.php 中 使 用 的 表单 元 素 如 表 2.8 所 示 。 


名 称 


forml 


bccdname 


text 


表 2.8 ”编辑 编程 词典 页 中 使 用 的 重要 表单 元 素 


重要 属性 
method="post" action="savebccdbb.php" onSubmit="retum 
chkinput(this)"> 
class="txt_grey" disabled="disabled" value=" 
<?php 
$sql4=mysql_query("select bccdname from tb_bccd where id= 
™.$_GET[bccdid]."",$conn); 
Sinfo4=mysql_fetch_array($sql4); 
echo unhtml(S$info4[bccdname]): ?> 


含义 
编辑 编程 词典 表单 


编程 词典 名 称 , 这 里 设置 
了 文本 的 只 读 属性 


bccdid 


bbid 


hidden 


Select 


value="<?php echo$_ GET id]:?> 
<?php $sql3=mysql_query("select * from tb_bb order by 
createtime desc ",$conn); 
S$info3=mysql_fetch_array($sq13); 
if($info3 二 false){ echo "<option> 暂 无 版 本 信息 </option>"; 
}else{ 
dof ?> 
<option value="<?php echo S$info3[id];?>"><?php echo 
unhtml($info3[bbname]);?></option> 
?php }while($info3=mysql_fetch_array($sql3)); }7> 


编程 词典 的 ID 


选择 编程 词典 的 版 本 


Submit 


Submit 


value=" 添 加 " class="btn_grey" 


提交 表单 
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sacvbccdbb.php 文件 对 表单 提交 的 数据 进行 处 理 , 首先 获取 表单 中 提交 的 数据 , 然后 判断 指定 的 版 
本 是 否 已 经 被 添加 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 代 码 如 下 : 
倒 程 23 ”代码 位 置 ， 光盘 \TM\02\bcty365\admin\savebccdbb.php 


<?php 
$bccdid=$_POST[bccdid]; // 获 取 表 单 中 提交 的 数据 
$bbid=$_POST[bbid]; /获取 编程 词典 ID 
$price=$_POSTI[price]; /获取 编 程 词典 单价 
$content=$_POST[content]; /| 获取 编程 词典 内 容 
$gn=$_POST[gn]; /| 获取 编 程 词典 功能 
$fw=$_POST[fw]; /获取 编程 词典 服务 
include_once("../conn/conn.php"); // 连 接 数 据 库 文件 
// 判 断 提交 的 编程 词典 是 否 已 经 被 添加 
$sql=mysql_query("select id from tb_bbqb where bccdid=".$bccdid."",$conn); 
$info=mysql_fetch_array($sql); /检索 指定 编程 词典 的 1D 
if($info!=falseX{ // 如 果 检 索 值 为 假 ， 则 弹出 提示 
echo "<script>alert(' 该 版 编程 词典 已 经 添加 !");history.back();</script>"; 
exit; } 


家 人 


$query=mysql_query("insert into tb_bbqb(bccdid,bbid,price,content,gn,fw) values('$bccdid','$bbid', '$price’, 


'$content','$gn','$fw')",$conn); // 将 表单 中 提交 的 数据 存储 到 数据 库 中 

// 更 新 编程 词典 的 价格 

$querys=mysql_query("update tb_bccd set bbid='$bbid',price='$price' where id=".$bccdid.""); 

if($query==true and $querys==true){ // 如 果 添 加 和 更 新 操作 为 真 ， 则 弹出 提示 
echo "<script>alert(' 版 本 信息 添加 成 功 !");history.back();</script>"; 

}else{ // 如 果 添加 和 更 新 操作 为 假 ， 则 弹出 提示 
echo "<script>alert(' 版 本 信息 添加 失败 !");history.back();</script>"; 

} 

?> 


2.14 软件 升级 管理 模块 设计 


2.14.1 软件 升级 管理 模块 概述 


软件 升级 管理 模块 实现 对 软件 升级 包 的 管理 ， 其 具体 的 功能 包括 添加 升级 包 、 编 辑 升级 包 、 添 加 
序列 号 和 编辑 序列 号 。 软 件 升级 管理 模块 中 的 添加 升级 包 和 添加 序列 号 是 一 一 对 应 的 ， 其 中 根据 所 属 
的 类 别 和 版 本 来 确定 升级 包 对 应 的 序列 号 ， 每 一 个 版 本 和 类 别 的 升级 包 对 应 一 个 序列 号 。 


2.14.2 ”软件 升级 管理 模块 技术 分 析 
在 软件 升级 包 管 理 模块 中 ， 应 用 到 一 个 动态 输出 下 拉 列 表 框 中 值 的 技术 。 下 面 就 来 讲解 一 下 该 技 
术 是 如 何 实现 的 ， 在 讲解 该 技术 之 前 ， 先 来 了 解 下 拉 列 表 框 的 基本 结构 。 
<select name="select"><!--name 指定 该 下 拉 列 表 框 的 名 称 -> 
<!--selected 设置 下 拉 列 表 框 的 默认 值 ， 默 认 值 为 PHP 一 > 
@ 


<option selected="selected">PHP</option> 
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<L-value 指定 的 mysql 是 下 拉 列 表 框 传递 的 值 ，MYSQL 为 显示 的 内 容 --> 
<option value="mysql">MYSQL</option> 
</select> 


所 谓 动态 输出 下 拉 列 表 框 中 的 值 就 是 从 数据 库 中 读 取 数据 , 将 获取 到 的 数据 输出 到 下 拉 
而 不 是 直接 在 下 拉 列 表 框 中 设置 某 个 固定 的 值 。 这 里 以 软件 升级 管理 模块 addsjb.php 文件 中 


列表 框 中 ， 
的 所 属 类 


别 下 拉 列 表 框 为 例 进行 讲解 ， 其 中 设置 下 拉 列 表 框 的 名 称 为 typeid， 默 认 值 为 “请 选择 ”，value 的 值 是 


从 数据 库 中 获取 的 ID 值 ， 显 示 的 内 容 为 从 数据 库 中 获取 的 类 型 名 称 。 动 态 输出 下 拉 列 表 框 中 
的 关键 代码 如 下 : 
倒 性 24 ”代码 位 置 ， 光盘 \TM\02\bcty365\ admin\wzdh.php 


h 的 值 使 用 


< 一 一 一 一 一 一 设置 下 拉 列 表 框 的 名 称 为 typeid- 一 一 一 一 一 一 一 一 一 一 一 一- - > 
<select name="typeid" class="txt_grey"> 
<option value="" selected="selected"> 请 选择 </option> 


< -一 一 一 一 一 一 一 一 一 一 设置 下 拉 列 表 框 的 名 称 为 ypeid- 一 -一 一 一 一 -一 一 -> 
<?php 

include_once("../conn/conn.php"); /| 连接 数据 库 

// 从 数据 库 中 读 取 编程 词典 类 型 的 数据 


$sql=mysql_query("select * from tb_type order by createtime desc",$conn); 
$info=mysql_fetch_array($sql); 
if($info==falseX{ 
echo "<option > 暂 无 类 别 </option>"; 
}else{ 
do 


echo "<option value=".$info[id].">".$info[typename]."</option>"; 


} 
while($info=mysql_fetch_array($sql)); 。“”//do…while 循环 语句 结束 


和 
</select> 


{ /应 用 do…while 循环 语句 输出 类 型 的 ID 和 类 型 的 名 称 


下 拉 列 表 框 不 但 可 以 动态 输出 数据 库 中 某 个 字段 的 数据 ， 而 且 可 以 输出 数组 中 的 数据 。 


下 面 就 实 


现 一 个 在 下 拉 列 表 框 中 动态 输出 数组 中 数据 的 功能 ， 首 先 创建 一 个 下 拉 列 表 框 ， 然 后 设置 下 拉 列 表 框 


的 值 ， 从 数组 中 读 取 数据 ， 应 用 for 循环 语句 进行 输出 。 代 码 如 下 : 


< 上 一 设置 下 拉 列 表 框 的 名 称 为 select 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 - 一 > 
<select name="select" size="1"> 
<?php 
$string="ASP,PHP.,JSP,.NET,DEL,VB,VC"; // 定 义 一 个 字符 串 
$srtings=split(",", $string); // 对 字符 串 进行 分 割 
$count=count($srtings); // 获 取 数 组 中 元 素 的 数量 
for($i=0;$i<$count;$i++){ /根据 数组 中 元 素 的 数量 进行 循环 输出 
$result=$srtings[$i]; /定义 变量 ， 获 取 数 组 中 指定 的 元 素 
echo "<option>$result</option>";} /将 数组 中 的 元 素 输出 到 下 拉 列 表 框 中 
?> 
</select> 


动态 输出 数据 库 中 的 数据 到 下 拉 列 表 框 的 运行 结果 如 图 2.48 所 示 。 
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的 文件 下 ， 关 


图 2.48 动态 输出 数据 库 中 的 数据 到 下 拉 列 表 杠 
2.14.3 ”软件 升级 包 上 传 的 实现 过 程 


国 ”软件 升级 包 上 传 使 用 的 数据 表 : tb_bb、 tb type、 tb_sjxz 
软件 升级 包 上 传 在 添加 升级 包 模块 中 实现 ， 通 过 一 个 文件 域 文本 框 将 升级 包 提交 到 服务 器 中 指定 


F 且 将 该 文件 在 服务 器 中 的 路 径 存 储 到 数据 


库 中 ,便于 在 前 台 实 现 对 软件 升级 包 的 下 载 。 其 运行 结 
果 如 图 2.49 所 示 。 

在 本 模块 中 通过 addsjb.php 文件 来 提交 升级 包 的 信 
息 , 通过 savesj.php 文件 来 对 表单 提交 的 数据 进行 处 理 。 
其 中 在 将 升级 包 上 传 到 服务 器 的 指定 文件 夹 的 过 程 中 ， 


主要 应 用 的 是 move_uploaded_file() 函 数 。 在 savesj.php 


文件 中 ， 首 先 获 取 表 单 提交 的 数据 ， 然 后 判断 服务 器 中 
是 否 存在 指定 的 文件 ， 最 后 应 用 move_uploaded_file0 
函数 将 升级 包 上 传 到 指定 的 文件 夹 下 ， 并 且 将 数据 存储 到 指定 的 数据 表 中 。 程 序 代码 如 下 : 

倒 程 25 ”代码 位 置 ， 光盘 \TM\02\bcty365\admin\savesj.php 


升 红包 名称 : 
所 民 织 别 
版 本 名 际 : 

下 二 站 过: 


说明 及 简介 : | 


图 2.49 软件 升级 包 上 传 的 运行 结果 


<?php 


$name=$_POSTIname]; 
$typeid=$_POST[typeid]; 
$content=$_POST[content]; 
$addtime=date("Y-m-j H:i:s"); 
$bbid=$_POST[bbid]; 
if(is_dir("./sjxz")==falseX{ 
mkdir("./sjxz"); 


’ 
S$link=date("YmjHis"); 
$path=$link.mt_rand(1000000,9999999).strstr($_FILES["address"]["name"],"");// 重 新 设置 升级 包 名 称 
$address="./sjxz/". $path; 
move_uploaded_file($_FILES["address"]["tmp_name"],$address); ”// 将 升级 包 上 传 到 指定 的 路 径 下 
$address="./admin/sjxz/".$path; 
include_once("../conn/conn.php"); 
// 将 上 传 的 数据 存储 到 数据 库 中 ， 这 里 将 升级 包 在 服务 器 中 的 路 径 存 储 到 数据 库 中 
$query=mysql_query("insert into tb_sjxz(name ,typeid,content,addtime,address,bbid) 
values('$name,','$typeid','$content','$addtime','$address','$bbid')",$conn); 
if($queryX{ 
echo "<script>alert( 升 级 包 添加 成 功 !");history.back();</script>"; 


}else{ 
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/获取 表单 提交 的 数据 

/获取 表单 提交 的 数据 

/获取 表单 提交 的 数据 

/定义 时 间 变量 

/获取 表单 提交 的 数据 

// 浏 断 指 定 的 文件 夹 是 否 存在 

// 如 果 指定 的 文件 夹 不 存在 ， 则 创建 一 个 指定 的 文件 夹 


// 获 取 一 个 时 间 
// 设 置 升级 包 在 服务 器 中 存储 的 指定 路 径 


// 获 取 升 级 包 在 服务 器 中 的 存储 路 径 
// 连 接 数据 库 文件 


// 如 果 添 加 操作 成 功 ， 则 弹出 提示 


// 和 如果 添 加 操作 失败 ， 则 弹出 提示 
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echo "<script>alert(' 升 级 包 添 加 失败 !");history.back();</script>"; 


} 
Ts 


2.14.4 软件 升级 包 删 除 的 实现 过 程 


国 软件 升级 包 删除 使 用 的 数据 表 : tb bb、tb type、tb sjxz 
软件 升级 包 删 除 的 实现 主要 根据 当前 数据 中 提供 的 ID, 执行 delete 删除 语句 ， 将 数据 表 中 相同 ID 
的 数据 删除 。 其 运行 结果 如 图 2.50 所 示 。 


全 最 和 中心 相 柜 : 000t-a918581 Carmiet 2001 区) 下 杆 省 明和 虹 人 司 


2.50 ”软件 升级 包 删 除 的 运行 结果 
该 功能 主要 通过 editsjb.php 文件 和 deletesjb.php 文件 实现 。 通 过 editsjb.php 文件 输出 数据 库 中 存储 
的 有 关 升 级 包 的 信息 ， 以 分 页 的 形式 显示 ， 在 每 条 记录 的 最 后 设置 一 个 删除 链接 ， 通 过 脚本 来 调用 
deletesjb.php 文件 ， 根 据 变量 中 的 ID 值 执行 删除 升级 包 的 操作 。 关 键 代码 如 下 : 
倒 程 26 ”代码 位 置 ， 光盘 \TM\02\bcty365\admin\deletesjb.php 


<?php 
$id=$_GET[id]; // 获 取 变 量 传递 的 ID 
include_once("../conn/conn.php"); /| 连接 数据 库 


// 执 行 删除 操作 ， 将 数据 表 中 对 应 的 ID 数据 删除 
if(mysql_query("delete from tb_sjxz where id=".$id."",$conn)X{ 
echo "<script>alert(' 该 升级 包 删 除 成 功 !");history.back();</script>"; ”// 如 果 删 除 操作 成 功 ， 则 弹出 提示 
}else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 
echo "<script>alert(' 该 升级 包 删除 失败 !);history back();</script>"; 


} 
?> 


2.15 在 Linux 系统 下 发 布 网 站 


在 Linux 系统 下 发 布 基于 PHP 的 网 站 ， 首 先 需要 配置 PHP 的 运行 环境 ， 其 次 需要 对 网 卡 参 数 进行 
设 定 。 这 里 将 以 发 布 “BCTY365 网 上 社区 ”网 站 为 例 讲解 在 Linux 系统 下 如 何 实现 网 站 的 发 布 。 假 设 
已 经 申请 到 表 2.9 所 示 的 网 络 参 数 。 
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表 2.9 申请 到 的 网 络 参 数 


参数 值 
IP 192.168.1.* 
Netmask 255.255.255.* 
Network 192.168.1.0 
Broadcast 192.168.1.* 
Gateway, 192.168.1.* 
主机 名 Tsoft 
DNS 168.95.1* 


在 Linux 系统 下 网 站 发 布 的 操作 步骤 如 下 : 
(1) 配置 PHP 的 运行 环境 ， 在 2.4 节 中 已 经 做 了 详细 介绍 ， 这 里 不 再 袭 述 。 
(2) 将 “BCTY365 网 上 社区 ”网 站 的 所 有 文件 复制 到 Apache 主 目 录 下 。 
(3) 设置 主机 名 称 。 在 终端 窗口 中 输入 如 下 命令 编辑 /etc/sysconfig/network 文件 : 


vi /etc/sysconfig/network 


将 该 文件 中 的 参数 NETWORKING 设置 为 yes, 表示 启动 网 络 , 将 参数 HOSTNAME 设置 为 Tsoft， 
表示 设置 主机 名 为 Tsoft。 
(4) 设置 网 卡 参数 。 在 终端 窗口 中 输入 如 下 命令 编辑 文件 /etc/sysconfig/network-scripts/ifefg-eth0: 


Vi /etc/sysconfig/network-scripts/ifcfg-ethO 


该 文件 的 相关 参数 设置 如 表 2.10 所 示 。 
表 2.10 设置 网 卡 的 相关 参数 


参数 说 了 明 
DEVICE=eth0 设置 网 卡 名 称 ， 要 与 这 全 -eth0 对 应 
ONBOOT=yes 指定 在 开机 时 启动 网 卡 
BOOTPROTO=static 设 定 启动 时 获取 IP 的 方式 
IPADDR=192.168.1.* 设 定 服务 器 IP 地 址 
NETMASK=255.255.255.* 设 定子 网 掩 码 
BROADCAST=192.168.1.* 设 定 同 网 段 的 广播 地 址 
GATEWAY=192.168.1.* 设 定 网 卡 的 网 关 


(5) 设置 DNS 主机 的 他 。 在 终端 编辑 /etc/resolv.conf 文件 : 
vi /etc/resolv.conf 


设置 参数 nameserver 的 值 为 168.95.1.*。 
(6) 重新 启动 网 络 设置 。 在 终端 窗口 中 输入 如 下 命令 : 


letc/rc.d/inin.d/network restart 
ifdow eth0 
ifup eth0 


i 
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(7) 打 开 浏 览 器 , 在 地 址 栏 中 输入 服务 器 IP 地 址 或 域名 , 打开 如 图 2.51 所 示 页 面 , 则 说 明 在 Linux 
系统 下 “BCTY365 网 上 社区 ”网 站 发 布 成 功 。 


ET 


FREE 5 7 
图 2.51 在 Linux 系统 下 “BCTY365 网 上 社区 ”网 站 运行 结果 


2.16 ”开发 技巧 与 难点 分 析 


2.16.1 ”管理 员 权限 的 设置 


为 了 更 好 地 管理 和 维护 网 站 的 论坛 ， 针 对 论坛 设置 了 一 个 管理 员 ， 该 管理 员 不 在 后 台 进 行 操作 ， 
而 是 在 前 台 为 管理 员 设置 特殊 的 权限 ， 也 可 以 称 之 为 版 主 。 其 水 现 的 后 理 是 ， 首先 在 数据 库 中 设置 不 
同 的 值 代表 不 同 的 权限 ，0 代表 普通 会 员 ，1 代表 管理 员 ; 然后 在 论坛 的 页 面 中 进行 判断 ， 当 用 户 的 类 


安 贝 ， EF- 
型 为 1 时 ， 不 但 具有 普通 会 员 的 权限 ， 而 且 具 有 删除 发 布 帖子 、 回 复 帖 子 和 项 帖 的 权限 ， 如 果 用 户 的 
类 型 不 是 1， 则 不 具有 上 述 的 权限 ， 只 能 是 发 布 和 回复 帖子 。 管 理 员 和 普通 会 员 登 录 的 页 面 效果 是 不 同 
的 ， 如 图 2.52 和 图 2.53 所 示 。 


CTETTIGI 
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[7 。 @c 


目 吉 主 网 :PER 于 夺 


im 存放 委 区 和 认 玉 开凿 全 手 和 中兴 和 全 


图 2.52 管理 员 登 录 的 操作 页 面 


图 2.53 普通 会 员 登 录 的 操作 页 面 
在 页 面 中 执行 的 判断 语句 判断 登录 用 户 的 类 型 ， 然 后 根据 类 型 判断 用 户 的 权限 。 程 序 关键 代码 如 下 : 
<?php 
if($_SESSION["unc"]!=™"X /| 判断 session 变量 的 值 是 否 为 空 
// 根 据 session 变量 的 值 获取 该 用 户 的 类 型 


$sqlu=mysql_query("select usertype from tb_user where usermc=".$_SESSION[T"unc"]."",$connm) 


$infou=mysql_fetch_array($sqlu); 
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if($infou["usertype"]==1}{ /| 判断 当 该 用 户 的 类 型 等 于 1 时 执行 下 面 的 内 容 
<!-- 调 用 JavaScript 脚本 ， 执 行 删除 发 布 帖 子 的 操作 --> 
<img src="images/lt_15(10).jpg" onclick="javascriptif(window.confirm(' 您 确定 删除 该 帖 么 ? 
')==true){window.location.href='bbs_delete.php?id=<?php echo $infob["id"]?>";}" style="cursor:hand"/> 
<?php 


, 
// 如 果 用 户 的 类 型 不 是 1， 则 不 执行 上 述 的 内 容 


?> 


2.16.2 ”帖子 置顶 的 设置 


所 谓 帖子 置顶 就 是 将 某 个 指定 的 帖子 在 对 应 的 版 块 中 最 前 面 的 位 置 显示 ， 该 权限 只 有 管理 员 才 拥 
普通 会 员 不 具备 该 权限 。 其 实现 的 原理 如 下 : 
首先 ， 在 数据 库存 储 发 布 帖子 信息 的 数据 表 中 设置 一 个 字段 ttpp， 指 定 该 字段 为 数字 类 型 ， 其 默认 
值 为 0。 

然后 ， 在 网 页 中 判断 登录 用 户 的 权限 ， 如 果 是 管理 员 ， 则 具有 帖子 置顶 的 权限 ， 否 则 将 弹出 提示 
对 话 框 “对 不 起 ， 您 不 具备 该 操作 权限 !”。 

最 后 ， 如 果 是 管理 员 ， 则 执行 settop.php 文件 ， 根 据 对 应 帖子 的 ID 查找 到 发 布 帖子 信息 表 中 对 应 
的 数据 ， 更 新 该 条 数据 中 top 字段 的 值 为 1。 

判断 登录 用 户 权 限 使 用 的 代码 如 下 : 


有 


<?php 
if($_SESSION["unc"]=="", // 判 断 session 变量 的 值 是 否 为 空 ， 如 果 为 空 则 执行 下 面 的 脚本 
echo "javascript:alert(' 请 先 登 录 本 站 ， 然 后 进行 此 操作 !");window.location.href='index.php';"; 
jelse{ 
// 如 果 不 为 空 则 执行 下 面 的 内 容 ， 判 断 登录 用 户 的 权限 
$sqlu=mysql_query("select usertype from tb_user where usermmc=".$_SESSION["unc"]."",$conn); 
$infou=mysql_fetch_array($sqlu); 
if($infou["usertype"]==1X{ 1/ 如果 登录 用 户 的 类 型 是 1， 则 说 明 是 管理 员 ， 执 行 下 面 的 内 容 
echo "javascript:window.location.href='settop.php?id=". $infob["id"].™"; 
}else{ 
1/ 如果 登 录用 户 不 是 管理 员 ， 则 执行 下 面 的 内 容 
echo "javascript:alert( 对 不 起 ， 您 不 具备 该 操作 权限 !"");"; 


} 
?> 
实现 帖子 置顶 是 通过 settop.php 文件 来 完成 的 , 在 该 文件 中 , 根据 变量 提交 的 值 获取 到 发 布 帖子 信 
息 表 中 对 应 的 数据 ， 更 新 该 条 数据 中 字段 top 的 值 ， 并 且 对 该 字段 的 值 进行 判断 。 如 果 字段 top 的 值 为 
1， 则 说 明 该 帖 已 经 置 项 ， 此 时 将 字段 的 值 更 新 为 0， 即 取消 置顶 ， 如 果 字 段 top 的 值 为 0， 则 说 明 该 
帖 没 有 进行 置顶， 此 时 将 字段 的 值 更 新 为 1， 即 置顶 该 帖 。settop.php 文件 的 程序 代码 如 下 : 


<?php include_once("conn/conn.php"); /| 连接 数据 库 文件 
// 根 据 获取 的 ID 值 ， 从 数据 表 中 读 取 到 对 应 的 数据 
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$sql=mysql_query("select top from tb_bbs where id=".$_GETT"id"]."™",$conn); 
$info=mysql_fetch_array($sql); 
if($info[top]==1){ // 判 断 对 应 数据 记录 中 字段 top 的 值 ， 如 果 字 段 top 的 值 为 1， 则 执行 下 面 的 内 容 


mysql_query("update tb_bbs set top=0 where id=".$_GET[T"id"]."",$conn);// 更 新 字段 top 的 值 为 0 
}elseif($info[topl]==0){ ””// 如 果 对 应 数据 记录 中 字段 top 的 值 为 0， 则 执行 下 面 的 内 容 
mysql_query("update tb_bbs set top=1 where id=".$_GET["id]."",$conn);// 更 新 字段 top 的 值 为 1 


echo "<script>alert(' 置 项 设置 成 功 !);history.back();</script>"; 
?> 


2.17 在 线 支付 技术 专题 


所 谓 在 线 支付 就 是 客户 端 〈 人 金融 机 构 需 客户 端 安装 由 金融 机 构 签发 的 数字 证 书 ， 信 用 卡 免 安 装 ) 
将 支付 信息 加 密 后 通过 互联 网 传送 到 支付 网 关 〈 支 付 网 关 是 解决 网 络 上 安全 支付 问题 的 交易 平台 ， 位 
于 互联 网 和 传统 的 金融 机 构 内 部 网 之 间 ， 其 主要 作用 是 将 互联 网 和 金融 网 络 安全 地 连接 起 来 ， 将 不 安 
全 的 网 上 交易 信息 传 给 安全 的 金融 网 络 ， 起 到 隔离 和 保护 金融 网 络 的 作用 )， 同 时 金融 机 构 网 上 支付 系 
统 反 馈 有 关 支 付 信息 ， 客 户 确认 无 误 后 进行 支付 确定 ， 支 付 网 关 负 责 商户 网 上 交易 资金 的 清算 ， 并 根 
据 商户 提供 的 开户 行 、 账 号 等 结账 信息 将 网 上 消费 款项 汇总 划 入 商户 账户 。 

BCTY365 网 上 社区 的 在 线 支付 是 与 中 国 工商 银行 合作 来 共同 完成 的 。BCTY365 网 上 社区 的 在 线 
支付 操作 步骤 如 下 : 

(1) 登录 网 上 社区 ， 如 图 2.54 
所 示 。 

(2) 购买 商品 。 在 本 页 中 , 不 但 
可 以 购买 商品 ， 还 可 以 查看 商品 的 
详细 信息 和 购物 车 中 的 商品 信息 ， 如 
图 2.55 所 示 。 洪 有 搞 在 司 内 工 个 与 页 三 寺 # 十 第 ! 页 / 共 1 页 首页 上 一 页 下 一 页 尾 页 

(3) 进入 购物 车 操作 页 面 。 在 该 
页 面 中 ， 可 以 修改 购物 数量 、 删 除 指 
定 商品 、 清 空 购物 车 、 继 续 购物 和 统计 购买 商品 的 金额 ， 也 可 以 单 击 “ 结 算 ” 按 钮 进入 到 商品 结算 页 
面 ， 如 图 2.56 所 示 。 
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图 2.54 在 线 订购 的 操作 页 面 
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价格 : 2,000.00 元 
是 版 。 权 : 吉林 省 明日 科技 有 限 公司 RCR RECT | i 
ED 人 到 | VE 得 词典 zoom | 下 ”更改 数 量 | 出 除 该 项 
全 4 继续 购买 滞 宇 购物 车 青 品 金额 总 计 : 2,000.00 元 [sw ] 
图 2.55 购买 商品 操作 页 面 2.56 ”购物 车 操作 页 面 


(4) 进入 到 购物 结算 页 面 ， 填 写 收 货 人 的 详细 信息 ， 确 认 后 提交 该 数据 ， 如 图 2.57 所 示 。 
(5) 订单 确认 。 订 单 确认 以 后 ， 就 可 以 提交 订单 ， 准 备 进行 网 上 支付 ， 如 图 2.58 所 示 。 
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末 李 生肖 村 有 限 司 -所 各 司 条 ETEE 


Wa 4 计 ( 元 》 


图 2.57 填写 收 货 人 的 详细 信息 图 2.58 订单 确认 
(6) 进行 网 上 支付 。 在 这 里 可 以 选择 工行 网 上 


支付 ， 也 可 以 选择 取消 该 订单 ， 如 图 2.59 所 示 。 ed ee 

接 下 来 的 操作 在 工行 B2C 支付 页 面 上 进行 。 首 : 
先 网 上 社区 按照 工商 银行 B2C 订单 数据 规范 形成 提 HD i | 
交 数 据 , 并 使 用 工商 银行 提供 的 API 和 商户 证 书 对 订 图 2.59 执行 网 上 支付 


单数 据 签 名 ， 形 成 form 表单 返回 客户 浏览 器 ， 表 单 
action 地 址 指向 工商 银行 接收 商户 B2C 订单 信息 的 servlet; 然后 在 客户 确认 使 用 工行 网 上 支付 后 ， 提 
交 此 表单 到 工商 银行 ， 最 后 工行 网 银 系统 接收 此 笔 B2C 订单 ， 对 订单 信息 和 商户 信息 进行 检查 ， 通 过 
检查 则 显示 工行 B2C 支付 页 面 。 

客户 通过 工行 B2C 支付 页 面 实 现 网 上 支付 ， 商 户 查询 网 上 银行 的 账户 ， 如 果 货 款 已 经 到 账 ， 则 根 
据 客户 指定 的 方式 将 货物 送 达 客户 手中 。 

上 述 内 容 就 是 网 上 社区 系统 的 在 线 支 付 流程 ， 涉 及 工商 银行 的 操作 内 容 这 里 不 做 讲解 。 这 里 主要 
讲解 一 下 如 何 将 订单 信息 提交 到 工商 银行 。 该 项 操作 主要 通过 shopping_tjdd.php 文件 来 实现 ， 首 先 从 
数据 库 中 读 取 订 单 信息 ， 然 后 将 订单 信息 进行 输出 ， 最 后 创建 “取消 订购 ”和 “工行 网 上 支付 ”两 个 
超 链接 ， 通 过 JavaScript 脚本 来 调用 不 同 的 执行 文件 。 关 键 代码 如 下 : 

<?php include_once("conn/conn.php"); include_once("top.php"); /连接 数据 库 和 网 站 的 头 文件 ?> 

<!-- 省 略 了 部 分 代码 --> 

<?phl 

© ee es ee 1/ 对 获取 的 订单 编号 进行 base64 解码 
// 获 取 该 订单 的 金额 信息 

$sql=mysql_query("select * from tb_dd where ddnumber=".$ddnumber."", $conn); 
$info=mysql_fetch_array($sql); 

$amount=$info["totalprice"]; 


@ $amount=str_replace(",",",number format($amount,2)); /修改 数字 的 输出 格式 
$amount=str_replace("."…,"",number_format($amount,2)); /修改 数字 的 输出 格式 
?> 

// 省 略 部 分 HTML 代码 


<table width="630" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr><td width="159">&nbsp; 
<?php 
$sql=mysql_query("select totalprice from tb_dd where ddnumber=".base64 decode($_GET ["ddno"]).""， 
$conn); 
$info=mysql_fetch_array($sql); 
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echo "<font color=red><strong>".$info["totalprice"]."&nbsp; 元 </strong></font>"; 
Fig </td> 
</tr> 
</table> 
5 // 省 略 部 分 HTML 代码 
<script language="javascript"> 
1/ 打印 订单 
© function openprintwindow(x,y,z}{ 
window.open("printwindow.php?ddno="+x+"&pv="+z,"newframe","top=200,left=200,width=635,hei 
ght="+(230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no,status=no"); 


} 
</script> 
je // 省 略 部 分 HTML 代码 
<table width="630" height="25" border="0" align="center cellpadding="0" cellspacing="0"> 
<tr> 
< 取消 该 订单 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 - -> 


<td width="75"><img src="images/bg_14(14).jpg” width="69" height="20" style="cursor:hand" onclick= 
"javascript:if(window.confirm(' 如 果 取 消 该 订单 ， 则 该 订单 将 被 删除 ， 您 需要 重新 购买 ! )==true){window. location. 
href='deletedd.php?ddno=<?php echo $_GETI["ddno"];?>";}"/></td> 
-一 一 执行 网 上 支付 一 一 一 一 -一 一 一 一 一 一 一 - > 
<td width="125"><img src="images/bg_14(15)jpg" width="119" height="20" onclick="javascriptwindow. 
location.href='ddform.php?orderid=<?php echo base64_decode($_GET["ddno"]);?>&amount=<?php echo $amount;? 
>&orderDate=<?php echo date("Ymdhis");?>';" style="cursor:hand"/></td> 


</tr> 
</table> 
< 一 一 一 一 省 略 了 部 分 代码 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 - -- > 
<?php include_once("bottom.php"); // 包 含 网 站 的 尾 文 件 ?> 
< 代码 贴 十 


@base64 decode(): PHP 实现 对 base64 编码 的 字符 进行 解码 。PHP 实现 字符 囊 的 base64 编码 通过 base64_encode() 
函数 。 
@ str_replace(): 实现 字符 串 的 替换 。 该 函数 的 语法 如 下 : 


mixed str_replace ( mixed search, mixed replace, mixed subject , int &count ) 


str_replace() 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替换 ， 参 数 &count 表 示 替 换 字符 串 执行 的 次 数 。 
@ openprintwindow 0: JavaScript 脚本 中 自 定义 的 函数 ， 用 于 执行 订单 的 打印 操作 。 


有 关 在 线 支付 流程 中 的 其 他 操作 实现 方式 已 经 在 2.10 节 中 进行 了 详细 的 讲解 ， 这 里 不 再 袭 述 。 其 
具体 的 代码 可 参考 本 书 的 光盘 TM\02\bcty365\。 


2.18 本 章 


本 章 从 项 目 开发 的 实际 角度 出 发 ， 以 某 科 技 公 司 的 实际 需求 为 背景 ， 详 细 地 讲解 “BCTY365 网 上 
社区 ”系统 的 开发 过 程 ， 其 中 以 系统 的 整体 开发 流程 为 主线 ， 重 点 介绍 技术 支持 、 在 线 订购 、 社 区 论 
坛 和 编程 词典 管理 等 儿 个 大 模块 的 实现 方法 ， 并 且 对 管理 员 权 限 设置 、 帖 子 置顶 设置 和 在 线 支付 技术 
进行 了 难点 分 析 和 专题 讲解 ， 而 且 在 本 章 中 还 讲解 了 在 Linux 系统 下 如 何 搭建 PHP 的 开发 环境 以 及 如 
何 发 布 网 站 。 
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办 公 自 动 化 管理 系统 


(Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


办 公 自 动 化 (Office Automation，OA) 系统 指 实 现 办 公 室 内 事 
务 性 业务 的 自动 化 。 

办 公 自 动 化 没有 明确 的 定义 ， 最 普遍 的 说 法 是 凡是 在 传统 的 办 公 
室 中 采用 各 种 新 技术 、 新 机 器 、 新 设备 从 事 办 公 上 业务 ， 都 属于 办 公 自 
动 化 的 领域 。 

办 公 自 动 化 系统 与 办 公 自 动 化 在 概念 上 存在 一 定 的 差别 。 办 公 自 
动 化 通常 指 办 公 室 中 配备 具有 自动 化 功能 的 设备 ， 这 些 设备 能 使 某 些 
办 公 活 动 自动 化 或 实现 某 个 单位 业务 的 自动 化 处 理 ; 而 办 公 自 动 化 系 
统 则 是 在 办 公 自 动 化 功能 的 基础 上 发 展 起 来 ， 以 办 公 自 动 化 技术 为 主 
体 ， 同 人、 组 织 、 制 度 、 环 境 等 相 结 合 的 完整 的 系统 。 通 过 阅读 本 章 
可 以 学 习 到 : 

办 公 自 动 化 管理 系统 的 开发 流程 

进一步 学 习 如 何 做 项 目 需求 分 析 与 系统 设计 
页 面 布局 中 的 框架 布局 

使 用 递归 图 数 做 多 级 下 拉 菜 单 
系统 日 志 的 实现 


于 


各 吾 吾 至 


潞 
ww 


办 公 自 动 化 管理 系统 ( Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 人 震 人 


31 开发 背景 


随 着 中 国 加 入 WTO 及 全 球 经 济 一 体 化 进程 的 加 快 ,世界 经 济 已 由 工业 化 经 济 逐 步 进入 网 络 信息 化 
时 代 。 在 信息 时 代 来 临 之 际 ， 各 企业 都 紧 跟 时 代 的 脚步 ， 转 变 着 企业 的 经 营 模式 、 管 理 模式 ， 从 传统 
的 人 工 管理 体制 向 信息 自动 化 管理 体制 过 渡 。 网 络 自 动 化 办 公 系 统 就 是 在 这 样 的 大 背景 下 应 运 而 生 的 。 

信息 时 代 的 到 来 让 人 们 尝 到 了 “信息 爆炸 ”的 滋味 ， 信 息 的 大 量 涌 入 让 企业 在 信息 处 理 方面 应 接 
不 暇 ， es et a hen eine ne A 
对 信息 数据 的 掌握 程度 、 处 理 能 力 ， 体 现 了 企业 对 市 场 的 敏感 程度 ， 数 据 的 真实 性 、 准 确 性 直接 决定 
着 企业 的 发 展 方向 。 从 传统 的 办 公 模式 向 自动化 办 公 管理 模式 转变 ， 提 高 企业 的 信息 处 理 能 力 ， 以 增 
强 企业 的 市 场 竞争 力 ， 成 了 企业 发 展 过 程 中 的 首要 问题 。 


3.2 需求 分 析 


根据 与 客户 的 多 次 交谈 和 了 解 ， 本 系统 所 面向 的 客户 对 象 情况 如 下 : 
所 属 IT 行业 ， 目 前 主要 以 开发 门户 网 、 企 业 网 等 中 小 型 网 站 为 主 。 
公司 经 过 多 年 经 营 ， 已 经 相对 稳定 ， 并 有 了 自己 的 网 站 、 企 标 、 规 章 制 度 和 基本 架构 。 
公司 规模 50 一 100 人 ， 人 和 手 一 机 ， 主 要 包括 部 门 经 理 、 普 通 职员 、 试 用 人 员 和 实习 人 员 。 
人 不 允许 访问 外 网 。 

公司 暂时 分 为 技术 部 、 人 事 部 和 质量 部 ， 但 不 排除 后 期 增加 其 他 部 门 的 可 能 。 

公司 实行 人 性 化 管理 ， 人 允许 员工 自由 发 表意 见 和 想法 ， 还 有 丰富 多 彩 的 活动 ， 如 比赛 、 旅 
游 等 。 
为 了 提高 工作 效率 ， 所 有 职员 定期 写 工作 计划 ， 包 括 周 计划 、 月 计划 ， 部 门 经 理 还 有 年 计划 
和 任务 计划 。 
根据 计划 的 完成 程度 和 完成 质量 ， 不 定期 地 选取 优秀 员工 。 


办 办 办 办 鸭 扫 


加 


加 


3.3 系统 设计 


3.3.1 系统 目标 


本 系统 是 针对 中 小 型 企业 内 部 自动 化 办 公 管 理 的 要 求 进行 设计 的 ， 主 要 实现 如 下 目标 : 
键盘 操作 ， 快 速 响应 。 

实现 文件 类 信息 的 强大 的 管理 能 力 。 

实现 对 员工 基础 信息 ‘人事 消息 ) 的 管理 功能 等 。 

实现 个 人 办 公 的 信息 自动 化 管理 功能 。 

发 布 会 议 信息 ， 并 对 会 议 信息 进行 管理 


加 ; 


回回 加 加 
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对 系统 用 户 进行 管理 。 

为 了 加 强 数 据 保密 性 ， 为 每 个 用 户 组 设置 权限 级 别 。 
系统 最 大 限度 地 实现 了 易 安装 性 、 易 维护 性 和 易 操作 性 。 
系统 运行 稳定 、 安 全 可 靠 。 


3.3.2 ”系统 功能 结构 


根据 系统 分 析 ， 下 面 给 出 系统 的 前 、 后 台 功 能 结构 图 。 
办 公 自 动 化 前 台 功 能 结构 图 如 图 3.1 所 示 。 


加 图 回回 


图 3.1 办 公 自 动 化 前 台 管理 系统 
办 公 自 动 化 后 台 功 能 结构 图 如 图 3.2 所 示 。 


3.3.3 系统 功能 预览 


办 公 自 动 化 管理 系统 由 多 个 功能 模块 组 成 ， 下 面 仅 列 出 儿 个 典型 功能 的 页 面 ， 其 他 页 面 参见 光盘 
中 的 源 程序 。 

前 台 登 录 界面 如 图 3.3 所 示 , 该 页 面 用 于 实现 对 用 户 登 录 的 用 户 名 和 密码 进行 验证 。 企业 信息 页 面 
如 图 3.4 所 示 ， 该 页 面 用 于 显示 企业 文化 和 各 规章 制度 。 
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噶 对 部 
一 只 粱 态 
工 需 串 


账号 权限 系统 日 志 
用 户 组 设置 数据 备份 
权限 分 配 修改 密码 


添 | | 修 | | 删 | | 查 
改 

职 | | 职 | | 职 | | 职 

员 | | 员 | | 员 | | 员 


图 3.2 办 公 自 动 化 后 台 管理 系统 


图 3.3 ”前台 登录 (光盘 \TM\03\oa\index.php) 图 3.4 ”企业 信息 光盘 \TM\03\oa\qyxx\r_system.php) 


个 人 计划 页 面 如 图 3.5 所 示 ， 该 页 面 主要 实现 用 户 的 工作 反馈 和 周 、 月 计划 等 功能 。 后 台 登 录 界面 
如 图 3.6 所 示 ， 该 页 面 用 于 管理 员 后 台 登 录 。 


Er 


图 3.5 个 人 计划 (光盘 \TM\03\oa\grjh\person_plan.php) 3.6 ”后 台 登 录 ( 光 盘 \TM\03\oa\admin\index.php) 
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职员 管理 页 面 如 图 3.7 所 示 ， 该 页 面 主要 用 于 实现 对 职员 的 查询 与 修改 等 功能 。 权 限 分 配 页 面 如 
图 3.8 所 示 ， 该 页 面 用 于 对 功能 使 用 权 的 分 配 。 


图 3.7 职员 管理 (光盘 \TM\O3\oavadmin\vzyglshow_stafphp) 图 3.8 权限 分 配 (光盘 \TM\03\oa\admin\qxgl\pur_assign.php) 


3.3.4 系统 流程 图 


用 户 在 登录 一 个 系统 后 会 进行 一 系列 的 操作 ， 把 这 些 操作 的 过 程 和 结果 以 图 形 的 形式 表现 出 来 ， 
这 就 是 系统 流程 图 。 一 个 好 的 流程 图 ， 不 仅 可 以 让 开发 者 迅速 地 理 清 思路 、 及 时 解决 出 现 的 问题 ， 也 
可 以 让 使 用 者 很 快 明白 该 系统 的 操作 方式 与 方法 。 办 公 自 动 化 管理 系统 的 工作 流程 图 如 图 3.9 所 示 。 


工作 组 


退出 系统 


图 3.9 办 公 自 动 化 管理 系统 流程 图 
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3.3.5 “开发 环境 


在 开发 办 公 自 动 化 管理 系统 平台 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows Server 2003 。 

服务 器 : Apache 2.2.8。 

PHP 软件 : PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.10.3。 
开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0。 

分 辩 率 : 最 佳 效果 为 1024x768 像素 。 


客户 端 

浏览 器 : 推荐 使 用 IE 6.0 及 以 上 版 本 。 
分 辩 率 : 最 佳 效果 为 1024x768 像素 。 
3.3.6 ”文件 夹 组 织 结构 


系统 功能 结构 和 系统 流程 设计 完成 后 , 接 下 来 设计 
网 站 的 文件 夹 结构 , 合理 的 文件 夹 结构 不 仅 易于 快速 地 
开发 ， 对 后 期 系统 的 调试 、 维 护 和 管理 也 能 起 到 事 半 功 
音 的 效果 。 本 系统 的 文件 夹 组 织 结构 如 图 3.10 所 示 。 


3.4 数据 库 设 计 


因 办 办 办 办 办 办 轨 


a 


加 回 


职员 管理 模块 

前 台数 据 库 性 接 文件 
前 台 Css 脚 本 文件 

个 人 计划 模块 

.前台 图 片 背景 目录 


审核 批示 模块 
职员 天 地 模块 


办 公 自 动 化 管理 系统 文件 夹 组 织 结构 


基于 目前 的 系统 开发 ， 没 有 数据 库 的 支持 根本 是 无 法 想象 的 ， 办 公 自 动 化 管理 系统 更 是 使 用 了 大 
量 的 数据 表 来 存储 数据 。 本 节 就 对 办 公 自 动 化 管理 系统 所 需要 使 用 的 数据 库 进 行 说 明 与 设计 。 


3.4.1 ”数据库 分 析 

由 于 本 系统 采用 的 是 PHP 语言 ， 数 据 库 理所当然 地 使 用 MySQL， 不 仅 是 因为 开发 成 本 低 ， 更 重 
要 的 是 两 者 之 间 的 默契 程度 和 稳定 程度 要 远 远 高 于 其 他 的 数据 库 组 合 ， 并 且 MySQL 数据 库 对 于 一 个 企 
业 的 内 部 办 公 自动 化 管理 系统 完全 够 用 。 
3.4.2 ”数据 库 概 念 设计 


根据 以 上 的 需求 分 析 及 系统 分 析 ， 规 划 出 办 公 自 动 化 管理 系统 主要 的 几 个 实体 关系 E-R 图 。 
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1. 用 户 信息 实体 
用 户 信息 实体 包括 职员 的 账号 、 密 码 、 姓 名 、 电 话 等 表示 个 人 身份 的 数据 资料 ， 如 图 3.11 所 示 。 
2. 考勤 登记 实体 


考勤 登记 实体 包括 登记 日 期 、 登 记 时 间 、 登 记 类 别 (上班 或 下 班 )、 登 记 状 态 (迟到 或 早退 和 登 
记 人 、 备 注 等 信息 ， 如 图 3.12 所 示 。 


图 3.11 用 户 信息 实体 E-R 图 图 3.12 考勤 登记 实体 E-R 图 
3.4.3 ”数据 库 物 理 结构 设 计 


在 本 系统 中 创建 了 一 个 数据 库 db_office， 一 共 包 含 13 个 
数据 表 ， 如 图 3.13 所 示 。 


器 服务 器 : localhost 部 数据 库 : db_office 
表 类 型 整理 


限于 篇 幅 ， 这 里 只 给 出 主要 的 表 结构 和 表 说 明 ， 其 他 数据 。 ee wen e302 ee 


表 结 构 请 参见 本 书 附带 的 光盘 。 也 _depart 。 WiSAM gb2312_chinese_cl 部门 管理 列表 
本 th_group MyISAM ”gb2312_chinese_ci 用户 姐 管理 列表 
tb iss MySAM gb2312_chinese_cl ”审核 列 表 
1. tb_users (用 户 列表 ) WA [Da3T2IERRGREIRI [9 度 


th MyISAM gb2312_chinese_ci 。 意见 箱 列表 


用 户 列表 主要 用 于 存储 职员 的 姓名 、 性 别 等 私人 信息 及 部 。 wparson EM Baiaaaal 人 有 衣 
门 、 工 作 组 等 与 公司 相关 的 公共 信息 。 该 数据 表 结 构 如 图 3.14 en ee 


tp_register 。 MySAM 。 gb2312_chinese_cl ”登记 列表 


所 示 。 tb_setup MWSAM gb2312_chinese_cl 时 间 列 表 
tb_superson MyISAM gb2312_chinese_ci 志和 员工 区 表 
2. tb_person (人 事 列 表 ) tb_users 。。 MYHSAM gb2312_chinese_ci 用 户 列表 


人 事 列表 用 于 存储 人 事 部 门 发 布 的 信息 ， 如 信息 标题 、 信 。 图 313 qb-office 数据 库 中 的 数据 表 列表 
息 内 容 、 信 息 时 间 等 。 该 数据 表 结构 如 图 3.15 所 示 。 


轩 服务 器 : localhost ”号 数据 库 : db_office ， 国 表 :tb_users 


字 恨 类 型 整理 屋 性 Nul 畦 认 往外 启明 

这 nt 否 auto_inerement 自动 六 号 

Muser varhar(50) 92312_chinese_ei 否 用 P 了 号 

upwd varthart20) 902312_chinese_ci 否 用 户 客 码 

uname 。 varchar(20) 902312chinese_cl 是 NULL 用 户 姓 名 

er 刁 汪 大 汪 殉 王 CE] 图 服务 器: ocalhost ， 加 数据库: db_office ， 国泰 :tb_person 

ubirth da 是 NULL 用 户 生日 有 类 型 E23 层 性 Hul 扶 认 祷 外 说 明 
uaddress varcharl50) 002312 chinese_ci 是 NULL 用 户 往 址 a ts) 和 auin_inerement 。 目 动 篇 导 
utel varshart20) gb2312_chin 是 NULL 用 户 电话 pie varchar(50) gb2312_chinese_ 否 公告 村 是 
u_email varcharts0) gb2312_chir 是 WULL 用 户 zmail Pcontert mediumtex 0hb2312_rhinese_G 理 公告 向 容 
udepart 。 varzhar(20) 否 所 属 部 门 plime -dae 理 发 布 时 间 
is_on inttD) 否 各 活 联 号 id ntis) 消息 关 别 

图 3.14 用 户 列表 图 3.15 人 事 列表 
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3.5 ”公共 模块 设计 


本 系统 的 公共 模块 包括 conn 数据 库 链接 文件 、css 样式 文件 、JavaScript 脚本 文件 以 及 inc 下 的 自 
定义 函数 文件 和 包含 文件 等 几 类 文件 , 其 中 数据 库 链接 文件 和 css 样式 文件 在 前 面 的 章节 中 有 过 系统 的 
介绍 ， 这 里 主要 讲解 本 系统 所 涉及 的 JavaScript 脚本 文件 和 部 分 自 定义 函数 文件 。 


3.5.1 JavaScript 脚本 


在 办 公 自 动 化 管理 系统 中 ，JavaScript 脚本 一 般 用 于 表单 元 素 验证 ， 如 判断 text 文本 框 输入 是 否 为 
空 ， 输 入 格式 是 否 符合 标准 等 。 在 网 页 中 使 用 JavaScript 脚本 的 方式 主要 有 以 下 3 种 : 
(1) 在 网 页 中 使 用 <script></script> 标 签 对 
<script></script> 标 签 对 可 以 放 在 网 页 的 任意 位 置 ， 一 般 放 在 <head></head> 或 <body></body> 之 间 。 
代码 如 下 : 
<head> 


<script>document.write(" 办 公 自 动 化 管理 系统 开发 ");</script> 
</head> 


(2) 在 单独 文件 中 使 用 

如 果 JavaScript 脚本 比较 多 ， 而 且 位 置 分 散 不 易 管 理 ， 可 以 统一 放 到 一 个 扩展 名 为 js 的 文件 中 ， 
使 该 文件 成 为 JavaScript 脚本 文件 。 在 脚本 文件 中 ,不 需要 使 用 <script></script> 标 签 对 ， 直 接 写 脚本 代 
码 即 可 。 当 有 页 面 需要 使 用 到 里 面 的 JavaScript 脚本 时 ， 可 以 这 样 引用 ， 代 码 如 下 : 


<script src="ad_js.js" jangauge="javascript"></script> 


(3) 在 表单 元 素 或 标签 中 使 用 
这 是 最 直接 的 使 用 方式 ， 如 果 是 少量 的 脚本 则 可 以 这 样 使 用 。 例 如 ， 在 超 链 接 标签 <a> 中 想 要 使 用 
JavaScript 脚本 ， 代 码 格式 如 下 : 


<a href="#" onclick="alert('hello');">hello</a> 


在 办 公 自 动 化 管理 系统 中 ， 这 3 种 方法 都 有 使 用 到 ， 在 后 面 涉及 具体 应 用 时 再 进行 说 明 。 下 面 先 
来 看 两 个 经 常 使 用 到 的 脚本 函数 。 
(1) 验证 函数 ， 用 于 判断 表单 元 素 是 否 为 空 。 如 果 为 空 ， 则 返回 包 lse， 并 将 光标 焦点 定位 到 出 问 
题 的 表单 元 素 。 程 序 代码 如 下 : 
倒 性 01 ”代码 位 置 : 光盘 \TM\03\oa\js\client_jsjs 


// 后 台 登 录 界 面 验证 脚本 
function check(X{ 
ilogin.usermame.value==""){ // 刊 断 用 户 名 是 否 为 空 
alert(" 请 输入 用 户 名 !); 
login.username.focus(); // 将 光标 焦点 定位 到 该 表单 元 素 


return false; 
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if(login.pwd.value=="" // 判 断 用 户 密码 是 否 为 空 
alert(" 请 输入 密码 !1"); // 如 果 为 空 ， 弹 出 提示 框 
login.pwd.focus(); /将 光标 焦点 定位 到 该 表单 元 素 
return false; // 返 回 false 


EO 
(2) 选择 框 函数 ， 用 于 删除 确认 。 当 要 执行 删除 操作 时 ， 使 用 JavaScript 脚本 文件 进行 确认 ， 以 
免 因为 误 操作 而 引起 不 必要 的 损失 。 
倒 性 02 代码 位 置 ， 光盘 VTM\WO3\VoaWjsvclient jsjs 


function cftm(){ 
说 confirm(' 确 认 要 删除 吗 ? // 选 择 框 函数 
return true; // 如 果 选 择 确认 ， 则 返回 true， 继 续 执行 
else 
return false; 1/ 否则 ， 返 回 false 
} 


3.5.2” 自 定义 函数 


在 inc 文 件 夹 下 ， 有 两 个 主要 文件 ，chec.php 文件 和 func.php 文件 。 其 中 ，chec.php 文件 是 权限 检 
查 文件 ， 是 办 公 自 动 化 管理 系统 每 个 页 面 都 要 引用 的 ，func.php 文件 是 自 定义 函数 文件 。 


1. chec.php 文 件 


对 办 公 自 动 化 管理 系统 来 说 ， 用 户 对 每 个 页 面 的 访问 都 要 有 该 页 的 权限 才 可 以 ， 如 果 没 有 权限 级 
别 的 限制 ， 随 意 访问 重要 的 资源 和 数据 ， 那 么 只 能 说 明 这 是 个 完全 失败 的 系统 。chec.php 文件 的 代码 
如 下 和 

倒 程 03 代码 位 置 ， 光盘 \TM\03\oa\inc\chec.php 

<?php 

session_start(); /开启 session 支持 

© ifllisset($_SESSION[u_name]) 

echo "<script>alert( 您 无 权 访问 );location='./index.php';</script>"; 
@ if$_SERVER['HTTP_REFERER] =="") 


echo "<script>alert( 本 系统 不 允许 从 地 址 栏 访问 ');history.go(-1);</script>"; 
?> 


< 代码 贴 十 

@ isset() 函 数 : 用 来 判断 session[u_name] 是 否 被 创建 ， 如 果 session 已 经 创建 ， 返回 true， 否 者 返回 false。 其 中 
Session[u_name] 是 用 户 登 录 验 证 通过 后 创建 的 ， 如 果 没 有 登录 的 步骤 或 session 超时 ， 那 么 就 会 提醒 用 户 没有 权限 。 

四 $_SERVER[HTTP_REFERER]: 系统 预定 义 变量 ， 存 储 的 是 上 一 页 的 URL 地 址 。 这 个 判断 的 作用 是 防止 当前 用 
户 访问 不 属于 自己 权限 内 的 网 页 。 


2. func.php 文件 
限于 篇 幅 ， 这 里 只 给 出 两 个 自 定义 函数 的 代码 和 说 明 ， 其 他 函数 请 到 光盘 中 自行 查看 。 


@ 
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(1) 读 取 字 段 函 数 read_field0 
该 函数 的 作用 是 根据 传 入 的 id 值 取得 相关 的 字段 名 ， 一 共有 4 个 参数 ， 函 数 代码 如 下 : 
倒 枉 04 ”代码 位 置 : 光盘 \TM\03\oa\inc\func.php 


function read_field($conn,$tablename,Sfieldname,$n_id)jf 
$sqlstr = "select ".$fieldname." from ".$tablename." where id = ".$n_id; /生成 SQL 语句 


$result = mysql_query($sqlstr$conn); /| 执行 SQL 语句 
$rows = mysql_fetch_row($result); // 返 回 结果 集 
return $rows[0]; // 返 回 需要 的 字段 名 


read_field0) 函 数 的 参数 说 明 如 表 3.1 所 示 。 
表 3.1 read field() 函 数 的 参数 说 明 


参数 名 称 参数 说 明 
S$conn 数据 库 链 接 资 源 变量 
Sfieldname 要 查找 的 字段 名 
Stablename 要 查找 的 表 名 

$n_id 要 查找 的 id 值 


(2) 处 理 消息 函数 re_message() 
该 函数 的 作用 是 返回 消息 结果 ， 一 共有 2 个 参数 ， 函 数 代 码 和 参数 解释 如 下 : 
倒 程 05 ”代码 位 置 ; 光盘 \TMW3\oavadminvinc\func.php 
function re_message($result, $l_addressX{ 
if($result) 


echo "<script>alert( 操作 成功 ! ');location=".$l_address.";</script>"; // 如 果 结果 为 true, 转 到 其 他 页 面 
else 


echo "<script>alert(' 系 统 繁忙 ， 请 稍 后 再 试 ');history.go(-1);</script>"; // 如 果 结果 为 false， 回 到 上 一 步 


. 


$result: 数据 库 返 回 结果 ,如果 返 回 值 为 true, 说 明 对 数据 库 的 操作 成 功 ; 如 果 返 回 值 为 false， 
说 明 操作 失败 。 
$1_address: 操作 成 功 后 ， 要 返回 的 URL 地 址 。 


3.6 前 台 首 页 设计 
对 于 一 个 办 公 自 动 化 管理 系统 来 说 ， 首 页 不 可 能 ， 也 不 允许 显示 太 多 的 内 容 和 复杂 的 设计 ， 因 为 
办 公 自 动 化 管理 系统 的 作用 是 帮助 提高 企业 的 管理 效率 和 职员 的 工作 效率 。 如 果 页 面 过 于 繁琐 ， 会 让 
职员 分 不 清 主 次 ， 找 不 到 重点 。 所 以 一 定 要 突出 重点 内 容 ， 显 示 关 键 功 能 。 
3.6.1 前 台 首 页 概述 


根据 用 户 对 各 个 功能 模块 的 使 用 频率 和 重要 程度 , 本 系统 的 首页 中 要 显示 的 模块 主要 有 以 下 3 部 分 : 


{27 
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回 ”网 站 首部 导航 栏 
回 ”网 站 左 侧 导 航 栏 
企业 信息 模块 : 
企业 绩效 模块 : 
人 事 消息 模块 : 
审核 批示 模块 : 
考勤 管理 模块 : 
个 人 计划 模块 : 
职员 天 地 模块 : 


> 


Vvvyvyv 


: 包括 当前 用 户 、 所 在 部 门 、 当 前 时 间 、 首 页 、 重 新 登录 和 退出 登录 。 
: 包括 各 个 管理 模块 及 分 类 。 


包括 公司 简介 、 规 章 制 度 、 组 织 结构 和 企业 管理 。 
包括 任务 绩效 、 质 量 绩效 和 绩效 评定 。 

包括 企业 公告 、 活 动 安排 和 消息 管理 。 

包括 发 布 审核 和 批示 审核 。 

包括 上 下 班 登 记 、 病 事假 登记 、 加 班 登记 和 考勤 设置 。 
包括 工作 反馈 、 周 计划 、 月 计划 、 年 计划 和 任务 计划 。 
包括 职员 浏览 、 意 见 箱 和 个 人 设 定 。 


回 ”网 站 主 显示 区 : 默认 显示 企业 公告 、 活 动 安排 、 个 人 计划 和 审核 批示 。 
本 案例 中 提供 的 前 台 首 页 如 图 3.16 所 示 。 该 首页 在 本 书 光盘 中 的 路 径 为 \TM\03\oa\pub_main.php。 


办 公 自 动 化 管理 系统 OA le 
rs 首部 导航 Ag 


TA 
GN 
NS 


eg Ex 


图 3.16 办公 自 动 化 管理 系统 前 台 首 页 


3.6.2 ”前 台 首 页 技术 分 析 


目前 网 页 页 面 的 布局 主要 有 两 种 方式 : 表格 布局 和 框架 布局 。 本 系统 前 台 页 面 的 布局 使 用 的 是 框 
架 布 局 。 框 架 布 局 可 以 将 容器 窗口 划分 为 若干 个 子 窗口 ， 每 个 子 窗口 可 以 分 别 显示 不 同 的 网 页 ， 网 页 
之 间 相 互 独立 ， 没 有 直接 的 关联 ， 又 由 一 个 网 页 将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 
者 的 浏览 器 中 。 框 架 布局 的 好 处 是 : 每 次 浏览 者 发 出 对 页 面 的 请 求 时 ， 只 下 载 发 生变 化 的 框架 页 面 ， 
其 他 子 页 面 保持 不 变 。 使 用 框架 可 以 非常 方便 地 完成 导航 工作 。 

框架 布局 的 使 用 格式 及 属性 介绍 如 下 。 


1. 框架 布局 格式 


框架 布局 的 格式 很 简单 ， 只 要 几 行 代码 即 可 ， 常 用 的 格式 如 下 : 


<html> 
<head> 


</head> 


<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 


& 
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<body> 


</body> 
</noframes> 
</html> 


其 中 <frameset> 和 <frame> 标 签 是 框架 集 标 记 ， 而 <noframes> 标 签 是 为 了 防止 浏览 器 不 支持 框架 而 


实行 的 一 种 补救 措施 。 如 果 浏 览 器 不 支持 框架 集 ， 就 会 执行 <noframes> 标 记 中 的 内 容 ， 让 用 户 能 够 正 
常 浏览 网 页 。 


2. 框架 集 属性 
框架 集 包含 各 个 框架 的 信息 ， 通 过 <frameset> 标 记 来 定义 。 框 架 是 按照 行 和 列 来 组 织 的 ， 可 以 使 用 
<frameset> 标 记 的 属性 对 框架 的 结构 进行 设置 。 下 面 给 出 框架 集 的 常用 属性 值 、 说 明和 应 用 举例 ， 如 
表 3.2 所 示 。 
表 3.2 ”框架 集 的 常用 属性 
属性 名 称 属性 说 明 应 用 举例 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， | <frameset cols="25%,100,*" > 
COLS 取 值 有 3 种 形式 : 像素 、 百 分 比 〈%) 和 | <frame></frame> 
相对 尺寸 (*) </frameset> 
<frameset rows="25%,100,*" > 
Raw 在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， <frame> 
取 值 和 COLS 类 似 ， 也 是 3 种 形式 <frame> 
Siframesetf> 
= framset cols="25%,*" cols="*" frameborder="0"> 
指定 框架 周围 是 否 显示 边框 , 取 值 为 1( 显 | ~ 
FRAMEBORDER | 示 边 框 , 默认 值 ) 或 0 (不 显示 边框 )》 | 
</frameset> 
指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 默 <framset cols="25%,*" cols="*" framespacing="1"> 
FRAMESPACING 认 是 无 间隔 的 
</frameset> 
<framset cols="25%,*" cols="*" frameborder="1" 
BORDE 指定 边框 的 宽度 ，frameborder 属性 为 1 时 | border="5"> 
该 属性 才 有 效 i 
</frameset> 
3. 框架 属性 


使 用 <frame> 标 记 可 以 设置 框架 的 属性 ， 包 括 框架 的 名 称 、 框 架 是 否 包含 滚动 条 以 及 在 框架 中 显示 
的 网 页 等 。<frame> 标 记 的 常用 属性 及 说 明 如 表 3.3 所 示 。 


表 3.3 ”框架 的 常用 属性 及 说 明 


指定 框架 的 名 称 


SRC 指定 在 框架 中 显示 的 网 页 文件 (包括 HTML、PHP、JSP 等 网 页 文件 ) 
FRAMEBODER 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框 ， 为 默认 ) 或 0( 不 显示 边框 
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续 表 


属性 说 明 
可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 
指定 框架 是 否 包含 滚动 条 。 属 性 值 可 以 是 yes (有 )、no (没有 ) 和 auto (自由 ) 


属性 名 称 
NORESIZE 
SCROLLING 


3.6.3 ”前台 首页 的 实现 过 程 


国 前台 首页 使 用 的 数据 表 : tb_users、tb_list、tb_person、tb_register、tb_plan 

办 公 自 动 化 管理 系统 的 前 台 首 页 采用 的 是 二 分 栏 结构 布 局 。 二 分 栏 布局 的 特点 是 简洁 、 大 气 、 个 
性 鲜明 ， 其 框架 设计 依据 其 内 容 形 式 的 变化 而 灵活 多 变 ， 结 构 简练 ， 非 常 符合 办 公 自 动 化 管理 系统 的 
风格 。 本 系统 中 框架 布局 的 具体 应 用 如 下 。 

倒 性 06 ”代码 位 置 : 光盘 \TM\03\oa\pub_main.php 


©@ <framesetrows="111,*" cols="*" frameborder="no" border="0" framespacing="0"> 
<frame src="top.php" name="topFrame” scrolling="No" noresize="noresize” id="topFrame" 
title="topFrame" /> 
<frameset rows="*" cols="220,*" framespacing="0" frameborder="not" border="0"> 
<frame src="left.php" name="leftFrame" scrolling="auto” noresize="noresize” id="leftFrame" 
title="leftFrame" /> 
<frame src="main.php" name="mainFrame" scrolling="auto" id="mainFrame" title="mainFrame" /> 
</frameset> 
</frameset> 
@ <noframes><body> 
</body> 
</noframes> 


Ah 代码 贴 二 
@ rows="111,*" cols="*": 定义 顶部 框架 。 
@ src="top.php": 定义 顶部 框架 内 显示 的 页 面 。 
图 rows="*" cols="220,*": 定义 下 部 左 侧 框架 。 
@ src="left.php": 定义 左 侧 框架 显示 的 页 面 。 
@ src="main.php": 定义 中 间 主 框架 显示 的 页 面 。 
@ <noframes>: 当 浏 览 器 不 支持 框架 集 时 显示 的 内 容 。 


3.7 人 事 消 息 模块 设计 


人 事 消息 模块 主要 是 对 文件 进行 收发 管理 ， 模 块 的 设计 和 实现 并 不 复杂 ， 但 在 整个 系统 中 的 位 置 
却 很 重要 。 因 为 公告 栏 和 意见 箱 面向 的 是 全 体 用 户 ， 企 业 的 最 新 动向 和 职员 的 意见 想法 首先 都 是 在 这 
里 体现 出 来 的 。 

3.7.1 人事 消息 模块 概述 


人 事 消息 模块 主要 包含 了 两 部 分 内 容 : 一 部 分 是 面向 全 体 用 户 的 ， 包 括 查 看 公告 、 活 动 、 意 见 箱 


@ 
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等 ， 另 一 部 分 是 仅 对 人 事 部 开放 ， 其 他 用 户 不 允许 、 也 不 会 看 到 的 页 面 ， 如 公告 管理 、 意 见 管理 等 。 


人 事 消 息 用 例 图 如 图 3.17 所 示 。 
O 
1 


一 ae (G2 
一 | 一 


| 
发 布 公告 查看 留言 ) |4 一 一 
六 用 记 | < }” 普通 用 户 


图 3.17 人 事 消息 用 例 图 
3.7.2 ”人 事 消息 模块 技术 分 析 


在 人 事 消息 模块 中 ,企业 公告 和 活动 安排 都 是 在 同一 显示 页 中 显示 (p_message.php), 为 了 区 分 不 
同 的 操作 ， 需 要 给 显示 页 (p_message.php) 传 一 个 u_id 值 ， 在 数据 表 中 根据 不 同 的 u_id 值 ， 取 得 不 同 
类 别 的 内 容 。 传 值 方式 有 url 传 值 方 式 〈 如 p_message.php?u_id=1 ) 和 表单 传 值 方式 。 

同样 地 ， 在 接收 页 中 ， 接 收 方式 也 有 两 种 : get 方式 和 post 方式 。PHP 中 使 用 预定 义 变量 $_GET 
和 $_POST 来 接收 传 值 ， 格 式 如 下 : 


$_GETIPOST[u_id]; /注意 ， 变 量 名 都 为 大 写 


对 于 unl 传 值 方 式 , 接收 页 始终 用 $_GET[] 变 量 来 接收 ,如果 使 用 form 表单 传 值 , 那么 就 要 看 form 
表单 中 method 属 性 的 设置 了 .如 果 method=get, 则 使 用 $_GETU] 变 量 ;如 果 method=post, 则 使 用 $_POSTD 
变量 。 

PHP 对 传 值 方式 和 接收 方式 规定 很 严格 ， 用 $_POST[u_id] 接 收 不 了 p_message.php?u_id=1 传 过 来 
的 参数 。 同 样 地 ， 用 $_GET[u_id] 也 接收 不 着 method=post 的 表单 元 素 值 。 


$6 注意 如 呆 在 phpini 配置 文件 中 GLOBAL-ON/OFF 这 行 的 值 为 ON, 那么 直接 写 Sname 就 可 以 
调用 表单 元 素 的 值 ， 而 不 区 分 get 和 post; 如 果 为 OFF， 则 不 可 以 。 直 接应 用 表单 名 称 十 分 方便 ， 
但 也 存在 着 安全 隐患 。 推 荐 关闭 GLOBAL。 

3.7.3 消息 管理 的 实现 过 程 


国 消息 管理 模块 使 用 的 数据 表 : tb_person 
单 击 “ 人 事 消息 ”选项 中 的 “消息 管理 ” 子 选 项 ， 在 主 框架 (mainFrame) 内 就 会 显示 消息 管理 页 
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面 ， 在 页 面 中 显示 发 布 过 的 消息 列表 和 对 消息 的 操作 ， 包 括 “ 修 改 ””“ 删 除 ” 和 “发 布 新 消息 ”。 消 息 
管理 页 面 的 运行 结果 如 图 3.18 所 示 。 


加 
Er [7 
mt1-t5 | 1B | 


图 3.18 消息 管理 页 面 运行 结果 


在 本 页 面 中 ， 系 统 首先 通过 权限 检查 文件 〈chec.php) 判断 用 户 是 否 为 登录 用 户 ， 是 否 为 非法 链接 。 
如 果 被 系统 判断 为 非法 操作 ， 系 统 将 直接 关闭 :如果 没 有 异常 情况 ， 则 系统 继续 执行 下 面 的 代码 在 后 
面 的 例 程 中 ， 几 乎 每 个 页 面 都 有 chec.php 文件 ， 如 果 不 是 特殊 说 明 ， 则 都 为 正常 登录 ， 到 时 将 不 再 具体 
说 明 )， 当 用 户 通过 检查 后 ， 系 统 将 访问 数据 库 , 从 人 事 消息 数据 表 中 (tb_person) 读 取出 消息 发 布 时 间 、 
消息 标题 ， 显 示 在 页 面 中 ， 并 通过 id 为 每 条 数据 设置 “修改 ”和 “删除 ”操作 。 程 序 关键 代码 如 下 : 
倒 程 07 ”代码 位 置 ， 光盘 \TM\03\oa\rsxx\p_manage.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; // 包 含 数据 库 连接 文件 

?> 

<link href="../css/style.css" rel="stylesheet" /> /1 引入 外 部 CSS 样式 文件 

<script src="../js/client_js.js"></script> /引入 js 文件 

<?php 

/* 显 示 全 部 消息 记录 */ 
$sqlstr = "select id,p_time,p_title from tb_person"; /生成 SQL 语句 
$result = mysql_query($sqlstr,$conn); /执行 SQL 语句 ， 生 成 结果 集 
while($rows = mysql_fetch_row($result)X{ /循环 输出 结果 集 记录 

echo "<tr>"; 


for($i=1;$i<count($rows);$i++){ 
echo "<td height=30 style='text-indent: 30px;>".$rows[$i]."</td>"; 

} 

echo "<td><a href='m_message.php?id=".$rows[0]."> 修 改 </a>/<a 
href='d_message_chk.php?id=".$rows[0]." onclick='return del_mess();'> 删 除 </a></td>"; 

echo "</tr>";} 
人 
?> 
<tr> 

<td height="30" align="right" valign="middle" colspan="3"><a href='add_manage.php' target="mainFrame"> 

发 布 新 消息 </a></td> 
</tr> 


他 
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下 面 分 别 介 绍 “ 发 布 新 消息 ””“ 修 改 ” 和 “删除 ”消息 的 实现 过 程 。 


1. 发 布 新 消息 
单 击 “ 发 布 新 消息 ” 超 链 接 ， 进 入 发 布 页 面 。 发 布 消息 页 面 的 运行 结果 如 图 3.19 所 示 。 


国 开 | 
图 3.19 ”发 布 消息 页 面 运 行 结果 
发 布 页 面 中 ， 主 要 包含 一 个 消息 表单 元 素 ， 部 分 表单 元 素 如 表 3.4 所 示 。 
表 3.4 发布 消息 页 面 的 主要 表单 元 素 


<option value="9"> 企 业 公告 </option> 
<option value="10"> 活 动 安排 <option> 
value=" 发 布 " onclick="return add_mess();" 


在 消息 页 面 中 输入 消息 的 相关 内 容 后 , 单 击 “发 布 按钮 , 系统 将 会 跳 到 消息 处 理 页 (add_manage_ 
chk.php) 进行 处 理 ， 将 消息 存储 到 数据 库 中 ， 然 后 再 返回 该 页 。 程 序 关键 代码 如 下 : 
倒 程 08 ”代码 位 置 ， 光盘 \TM\03\oa\rsxx\add_manage_chk.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
include "../inc/func.php"; /包含 函数 处 理 文件 


/添加 新 消息 */ 
S$sqlstr = "insert into tb_person values(",".$_POSTIp_tile].",".$_POSTIp_content]}.",now(),".$_POSTIp_type].")"; 


$result = mysql_query($sqlstr,$conn); 
re_message($result"p_manage.php"); // 消 息 处 理 函 数 


人 


?> 


2. 修改 消息 
消息 发 布 后 ， 可 以 随时 对 消息 内 容 进 行 修改 。 在 消息 管理 页 面 中 , 选择 要 修改 的 消息 列 ， 单 击 “ 修 
改 ” 超 链接 将 进入 消息 修改 页 面 。 在 消息 修改 页 面 中 ， 有 一 个 和 发 布 页 面 非常 相似 的 “修改 ”表单 。 


全 
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消息 修改 页 面 中 的 主要 表单 元 素 如 表 3.5 所 示 。 
表 3.5 修改 页 面 的 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 
addmess, form action="m message chk.php" method="post" 修改 消息 表单 
_ptitle text value="<?php echo $rows[1]; ?>" 消息 标题 
content textarea <2?php echo $rows[2]: ?> 消息 内 容 
<?phpif($rows[4] 一 "9"){?> 
<option value="9" selected="selected"> 企 业 公告 </option> 
<option value="10"> 活 动 安 排 </option> 
p_type select <?php}else{?> 消息 类 型 
<option value="9"> 企 业 公 告 </option> 
<option value="10" selected="selected"> 活 动 安排 </option> 
<?php}?> 
id hidden value="<?php echo $rows[0]: ?>" 修改 消息 id 
submit2 submit value=" 修 改 " onclick="return add_mess();" “修改 ”按钮 


消息 修改 完成 后 ， 单 击 “ 修 改 ” 按 钮 ， 在 m_message_chk.php 页 中 将 会 处 理 修改 事件 ， 通 过 隐藏 
表单 传 过 来 的 id， 定 位 到 要 修改 的 消息 项 ， 使 用 update 语句 实现 对 数据 的 更 新 。 程 序 关键 代 码 如 下 : 
倒 程 09 ”代码 位 置 ， 光盘 \TM\03\oa\rsxx\m_message_chk.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 链接 文件 
/更 新 数据 库 ?/ 


$sqlstr = "update tb_person set p_title = ".$_POST[p_title].",p_content = ".$_POSTI[p_contentl.",p_time = 
now(),u_id =".$_POSTIp_type]." where id = "$_POSTIid]; 
$result = mysql_query($sqlstr$conn); 


re_message($result"p_manage.php"); /消息 处 理 函 数 
J 
3. 删除 消息 办 公 自 动 化 管理 系统 QO) 人 


上 二 分 关 号 宁 开 


每 隔 一 段 时 间 管 理 员 就 要 定期 清理 一 芷 
下 无 用 的 消息 和 活动 安排 , 以 加 速 页 面 的 浏 


览 速度 ,减轻 服务 器 压力 。 删除 消息 的 运行 二 
结果 如 图 3.20 所 示 。 二 
删除 消息 时 ， 选 择 要 删除 的 消息 列 ， 单 二 二 


击 “删除 ” 超 链接 ， 弹 出 确认 删除 对 话 框 ， 
单 击 “确定 ”按钮 ， 将 要 删除 消息 的 id 号 
传 给 系统 ， 通 过 消息 处 理 函数 将 消息 删除 。 
关键 代码 如 下 : 


局 


图 3.20 删除 消息 的 运行 结果 
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倒 程 10 ”代码 位 置 : 光盘 \TM\03\oa\rsxx\d_message_chk.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 库 链 接 文件 
include "../inc/func.php"; /包含 函数 处 理 文件 
/删除 数据 六 


$sqlstr = "delete from tb_person where id = ".-$_GET[id]; 
Sresult = mysql_query($sqlstr$conn); 
re_message($result,"./p_manage.php"); // 消 息 处 理 函 数 


iain 


3.7.4 意见 箱 的 实现 过 程 


国 意见 箱 模块 使 用 的 数据 表 : tb lyb 

意见 箱 是 用 户 匿名 发 言 的 一 个 留言 板 。 对 于 用 户 的 留言 ， 只 有 人 事 部 的 人 员 才 有 权 回 复 或 删除 ， 
对 于 其 他 部 门 的 员工 则 只 可 以 留言 和 查看 ， 而 
无 法 进行 其 他 操作 。 意 见 箱 的 运行 结果 如 图 
3.21 所 示 。 

在 本 页 面 中 ， 系 统 首 先 读 取出 意见 箱 数据 
表 (tb_lyb) 中 的 所 有 数据 ， 并 逐条 显示 ， 在 每 
条 显示 记录 后 ， 有 一 个 判断 式 ， 判 断 用 户 所 属 mn 
部 门 是 否 为 人 事 部 ， 如 果 是 人 事 部 成 员 ， 将 显 Se 
示 “ 回 复 ” 和 “删除 ” 超 链接 如 果 不 是 人 事 
部 ， 则 不 显示 这 两 个 操作 。 这 里 是 强制 性 的 ， 
也 就 是 说 ， 即 使 是 管理 员 ， 也 无 法 设置 这 里 的 
权限 ， 只 有 开发 人 员 才 可 以 改变 。 如 果 想 使 用 
这 种 技术 ， 就 一 定 要 确保 这 项 功能 权限 不 会 轻 
易 地 更 改 。 意 见 箱 页 面 的 关键 代码 如 下 : 

倒 程 11 ”代码 位 置 ， 光盘 \TM\03\oa\zytd\lyb.php 


Cr 


加 到 
图 3.21 意见 箱 页 面 的 运行 结果 


<?php 

session_start(); /开启 session 支持 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文 件 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 
人 * 显 示意 见 箱 信 息 */ 


$l_sql = "select id,|_title,l_content,|_time,is_reply,r_back from tb_lyb order by id desc"'; 
$l_rst = mysql_query($l_sql,$conn); 


while($!_rows = mysql_fetch_array($l_rst)X{ // 开 始 循环 输出 每 条 记录 
学 
<tr> 
<td width="15%" height="30" align="left" valign="middle" scope="col"> 
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<?php echo $l_rowsll_time]; ?> 

</td> 

<td align="left" valign="middle" scope="col"> 

主题 : <?php echo $l|_rows[l_title]; ?> 

</td> 

<!-- 判断 登录 用 户 是 否 为 人 事 部 成 员 --> 
<td width="15%" align="center valign="middle" scope="col"> 
<?php if(($_SESSION[u_deparl) == "人 事 部 "){ ?> 

<!-- 当 登录 用 户 是 人 事 部 成 员 时 ， 显 示 可 执行 操作 --> 

<a href="t_back.php?id=<?php echo $I_rows[id]; ?>"> 回 复 </a>&nbsp;&nbsp;<a href="del_ly_chk.php?id= 

<?php echo $l_rows[id]; ?>" onclick="returm del_mess()"> 删 除 </a> 
<?php} 
?>&nbsp; 
</td> 
<1-- -一 一 一 一 一 一 -- > 
</tr><tr> <td height="75" colspa " align="|left" valign="middle" scope="col" > 

<?php echo $I_rowsl[l_content]; ?> 

</td></tr> 

<?php 

<!-- 当 字段 is_reply 的 值 为 1 时 ， 说 明 有 回复 信息 --> 
if($l_rows[is_reply] == 1){ 


?> 
<tr> 
<td height="25" colspan="3" align="left" valign="middle" headers="50" scope="col" 
onmouseover="this.style.backgroundColor=#FFEEBC" onmouseout="this. style.backgroundColor=""> 
<font color="#FF0000"> 
<!-- 显 示 回复 信息 一 > 
<?php echo $l_rows[r_back]; ?> 
</font></td> </tr> 
<?php}} 


下 


?></table> 


<!-- 留 言 菜单 --> 


< 一 一 一 一 一 一 一 一 一 一 一 > 


下 面 介 绍 匿名 留言 和 回复 、 删 除 留 言 的 实现 过 程 。 
1. 匿名 留言 
留言 页 面 主要 是 一 个 HTML 表单 ， 因 为 允许 匿名 留言 ， 所 以 表单 中 没有 隐藏 域 ， 该 页 面 中 的 表单 


元 素 列 表 如 表 3.6 所 示 。 


表 3.6 留言 页 面 的 菜单 表单 元 素 表 


称 | 元 素 类 型 重要 属性 


form id="lyb" name="lyb" method="post" action="lyb_chk.php" 


1 title text type="text" name="] title" size="30" 

1 content textarea | name="l content" cols="75" rows="10" 

Submitl hidden type="reset" value=" 重 置 " 

Submit submit type="submit" value=" 提 交 " onclick="retum add_lyb();" 
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当 用 户 填 写 完 意见 标题 和 意见 内 容 后 ， 单 击 “ 提 交 ” 按 钮 ， 发 布 意见 ， 系 统 将 转 到 意见 处 理 页 
〈lyb_chk.php ) 中 对 意见 进行 处 理 : 如 果 意 见 标题 和 意见 内 容 都 不 为 空 , 则 执行 insert 语句 , 向 表 tb lyb 
中 添加 一 条 新 数据 。 程 序 关键 代码 如 下 : 


<?php 
Session_start(); /开启 session 支持 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
include "Jinc/chec.php"; /包含 权限 检查 文件 
include "../inc/func.php"; /包含 自 定义 函数 文件 


/判断 输入 是 否 为 空 ， 如 果 为 空 ， 则 返回 ， 如 果 无 错误 ， 则 执行 insert 语句 添加 新 记录 */ 
if((tim($_POSTI| title]) !{= "") and (trim($_POST[L_ content) != ™")){ 
$l_sql = "insert into tb_lyb values(",".$_POSTI| title].",”".$_POSTII_content].",now(),",")"; 
$l_rst = mysql_query($I_sql,$conn); 


re_message($l_rst,"lyb.php?u_id=24"); // 使 用 自 定义 函数 
人 
} 
else 

echo "<script>alert(' 内 容 和 消息 不 允许 为 空 ');history.go(-1);</script>"; 
?> 


2. 回复 、 删 除 留言 操作 

在 意见 箱 页 面 中 ， 当 系统 判断 当前 登录 用 户 为 人 事 部 成 员 后 ， 将 显示 留言 信息 后 面 的 、 可 操作 的 
功能 模块 “回复 ”和 “删除 ”。 单 击 相应 留言 主题 后 面 的 “回复 ” 超 链接 ， 进 入 回复 留言 页 面 。 回 复 留 
言 页 面 的 运行 结果 如 图 3.22 所 示 。 


ms| 


图 3.22 回复 留言 页 面 的 运行 结果 
回复 留言 页 面 中 的 主要 表单 元 素 如 表 3.7 所 示 。 
表 3.7 回复 留言 表单 的 主要 元 素 


tback action="t_back_chk.php" method="post" 


ttitle | text | value="<?php echo read field(Sconn,"tb lyb","l title",$_GET[id]); ?>"” | 意见 主 

r back | textarea | name="r back" id="r back" cols="75" rows="10" 回复 内 容 
id hidden | name="id" value="<?php echo $_ GET[id]; ?>" 回复 消息 id 
submit2 submit value=" 回 复 " onclick="return re_ back0:" 


在 回复 留言 页 面 中 ， 当 用 户 输入 回复 信息 后 ， 单 击 “ 回 复 ” 按钮 提交 表单 ， 在 表单 处 理 页 (t_back_ 
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chk.php) 中 系统 将 会 根据 回复 信息 ， 更 新 表 记 录 。 关 键 代 码 如 下 : 
倒 程 12 ”代码 位 置 ， 光盘 \TM\03\oa\zytd\ t_back_chk.php 


<?php 

session_start(); /开启 session 支持 
include "../conn/conn.php"; /包含 数 据 库 链接 文件 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 


上 * 根 据 id， 更 新 数据 库 */ 

$sqlstr = "update tb_lyb set r_back=".$ POSTIr_back].",is_reply = 1 where id = ".$_POSTI[id]; 
$result = mysql_query($sqlstr,$conn); 

re_message($result,"lyb.php?u_id=24"); // 信 息 函 数 


人 


3 


对 于 不 需要 再 保留 的 留言 信息 ， 可 将 其 删除 。 单 击 相应 消息 主题 后 面 的 “删除 ” 超 链 接 ， 弹 出 确 
认 删 除 提示 对 话 框 ， 单 击 “确定 ”按钮 将 消息 的 id 传 给 系统 ， 通 过 SQL 语句 将 其 删除 。 程 序 关键 代码 
如 下 : 


倒 性 13 ”代码 位 置 ， 光盘 \TM\03\oa\zytd\del ly_chk.php 


<?php 

session_start(); /开启 session 支持 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 
$sqlstr = "delete from tb_lyb where id = ".$_GETI[id]; // 删 除 信息 的 SQL 语句 
$result = mysql_query($sqlstr,$conn); /执行 语句 
re_message(S$result,"lyb.php?u_id=24"); // 返 回 结果 

?> 


3.8 考勤 管理 模块 设计 


考勤 功能 是 办 公 自 动 化 管理 系统 中 每 天 都 要 使 用 到 的 功能 模块 之 一 ， 也 是 比较 重要 的 模块 之 一 。 
除了 包括 正常 的 上 下 班 登记 功能 外 ， 还 包括 病 事假 登记 和 加 班 登记 等 特殊 登记 功能 。 管 理 员 通过 考勤 
记录 实现 对 上 下 班 及 加 班 标准 时 间 的 设置 。 


3.8.1 考勤 管理 模块 概述 


考勤 模块 主要 的 功能 介绍 如 下 。 

上 下 班 登记 : 包括 上 班 登记 和 下 班 登记 。 

病 事假 登记 : 包括 病假 登记 和 事假 登记 。 

加 班 登记 : 包括 加 班 上 班 登记 和 加 班 下班 登 记 。 

考勤 记录 : 包括 上 下 班 标准 时 间 设 置 和 加 班 标准 时 间 设 置 。 
考勤 模块 的 活动 图 如 图 3.23 所 示 。 


@ 
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图 3.23 考勤 管理 活动 图 


3.8.2 考勤 管理 模块 技术 分 析 


在 PHP 中 查询 数据 表 中 数据 的 一 般 步 骤 如 下 : 

(1) 链接 数据 库 〈 链 接 文件 conn.php )。 

(2) 执行 SQL 语句 ， 返 回 查 询 结果 集 ($result = mysql_query($sqlstr,$conn)。 

(3) 配合 while 循环 语句 输出 查询 结果 (while($rows = mysql_fetch_row($result)))。 

(4) 关闭 数据 库 。 

在 步骤 (3) 中 ， 如 果 要 取得 某 个 字段 的 值 ， 使 用 $rows[$num] 即 可 。 但 通过 图 3.23， 读 者 可 以 发 
现 , 从 开始 登记 到 保存 登记 记录 ,虽然 只 是 一 个 功能 的 实现 , 却 要 进行 至 少 3 次 判断 ,如 果 都 用 “0,1,2…” 
做 数组 下 标 ， 过 一 段 时间 后 ， 根 本 记 不 起 取得 的 数据 是 什么 、 有 什么 作用 。 在 后 期 测试 和 系统 维护 时 ， 
还 要 逐一 地 查找 数据 表 来 比 对 。 除 了 mysql_fetch_row() 函 数 外 ，PHP 还 提供 了 其 他 相似 的 函数 ， 下 面 
就 来 学 习 一 下 。 

1. mysql_fetch_array() 函 数 


mysql_fetch_array(O) 函 数 的 用 法 和 mysql_fetch_row0 函 数 十 分 相似 ， 唯 一 不 同 的 是 结果 集 的 数组 下 
标 是 所 查找 的 数据 表 的 字段 值 。 
例如 ， 读 取 数 据 表 tb_register 中 的 “登记 时 间 ” 字 段 ， 代 码 如 下 : 


$str = $rows[r_ time]; 


这 样 ， 即 使 时 间 间 隔 比较 长 ， 甚 至 换 人 开发 ， 也 能 明白 该 变量 的 含义 。 
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2. mysql_fetch_object() 函 数 


mysql_fetch_object() 函 数 与 其 他 两 个 函数 的 最 大 不 同 就 在 于 :获取 的 结果 集 是 以 对 象 的 形式 存储 
的 ， 而 不 是 数组 的 形式 。 也 就 是 说 ， 在 读 取 字段 变量 时 所 使 用 的 格式 不 再 是 $rows[$num]， 代 码 如 下 : 


$str = $rows-->r_time; 
下 面 来 看 一 个 实例 ， 分 别 使 用 上 面 3 种 不 同 的 函数 来 获取 数据 表 tb_setup 的 id 值 ， 代 码 如 下 : 


<?phi 


p 

让 链接 数据 库 */ 

$conn = mysql_connect("localhost", "root","root"); 
mysql_select_db("db_office",$conn); 
mysql_query("set names gb2312"); 

让 创建 和 执行 SQL 语句 */ 

$sqlstr = "select * from tb_setup"; 

$result = mysql_query($sqlstr, $conn); 

/使 用 mysql_fetch_row() 函 数 输出 字段 值 / 

$rows_1 = mysql_fetch_row($result); 

$str = "mysql_fetch_rows 函数 : ".$rows_1[0]."<br>"; 
/使 用 mysql_fetch_array() 函 数 输出 字段 值 / 

$rows_2 = mysql_fetch_array($result); 

$str .="mysql_fetch_array 函数 : ".$rows_2[id]."<br>"; 
/使 用 mysql_retch_object() 函 数 输出 字段 值 */ 
S$rows_3 = mysql_fetch_object($result); 

$str .= "mysql_fetch_object 函数 : ".$rows_3->id."<br>"; 
echo "<script>alert(".$str.");</script>"; 

?> 


输出 结果 为 : 


mysql_fetch_rows 函数 : 1 
mysql_fetch_array 函数 : 2 
mysql_fetch_object 函数 : 3 


3.8.3 ”上 下 班 登记 的 实现 过 程 


国 。 上 下 班 登 录 使 用 的 数据 表 : tb_register 

单 击 “ 考 勤 管理 ”选项 中 的 “上 下 班 登记 ” 子 
选项 ,在 主 显示 区 (mainFrame) 内 显示 个 人 上 下 班 
的 登记 记录 和 登记 链接 ， 用 户 可 以 在 这 里 进行 登记 
和 查找 。 上 下 班 登记 页 面 的 运行 结果 如 图 3.24 所 示 。 于 一 = 

在 上 下 班 登 记 页 面 中 ， 通 过 登录 用 户 id， 从 数 图 3.24 上 下 班 登记 的 运行 结果 
据 表 tb_register 中 返回 用 户 的 登记 记录 集 , 并 显示 在 
页 面 中 。 在 显示 数据 时 ， 需 要 进行 2 次 判断 : 一 次 判断 用 户 的 登记 类 型 (上班 还 是 下 班 ); 另 一 次 判断 
用 户 的 登记 状态 (人 述 到 、 早 退 、 正 点 上 下 班 )。 关 键 代码 如 下 : 


@ 
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倒 程 14 ”代码 位 置 : 光盘 \TM\03\oa\kqgl\work_note.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 库 链 接 文件 
include "../inc/func.php"; /包含 自 定义 函数 文件 

Vy 

<link href="../css/style.css" rel="stylesheet" /> 1/ 引进 外 部 css 样式 文件 


<script src="../js/client_js.js"></script> /1/ 引 进 js 脚本 文件 


© <a href="#" onclick="javascript:openWin=open('p_login.php?r_id=<?php echo $_GET[u_ id]; ?>' ,", 
"width= 420,height=260,scrollbars=no'");"> 
<?php 
/* 判 断 是 哪 种 功能 类 型 */ 
@ if$_GET[u_id] == 14X 
echo "上 下 班 登记 "; 
Jelse if($_GET[u_id] == 15)f 
echo " 病 事假 登记 
Jelse if($_GET[u_id] == 16)f 
echo "加 班 登 记 "; 


} 
?></a> 
<?php 

让 显示 登记 信息 */ 

$sqlstr = "select id,r_date,r_time,r_type,r_state,r_remark from tb_register where r id = ".$_GET[u_id]." 
and p_id = ".$_SESSIONI[id]; 

$result = mysql_query($sqlstr, $conn); 

while($rows = mysql_fetch_array(S$result)X 


?> 
<tr> 
<!-- 输 出 登记 日 期 --> 
<td height="25" align="center" valign="middle"><?php echo $rows[r_date]; ?></td> 
<!-- 输 出 登记 时 间 --> 
<td height="25" align="center valign="middle"><?php echo $rows[r_time]; ?></td> 
<td height="25" align="center" valign="middle"> 
<!-- 输 出 登记 类 型 -> 
<?php 
目 switch($rows[r_type]){ 
case 0: 
echo “下 班 "; 
break; 
case 1: 
echo "上 班 "; 
break; 
Case 2: 
echo "加 班 签到 "; 
break; 
Case 3: 
echo "加 班 签 退 "; 
break; 
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case 4: 


echo "病假 "; 
break; 

case 5: 
echo "事假 "; 
break; 


}?></td> 
<!-- ——————————- -> 
<td height="25" align="center" valign="middle"> 
<!-- 输 出 登记 状态 -> 


<?php 


9 Switch($rows[r_type]){ 


j?> 


<| -- 一 一 一 一 一 一 


</td> 


case 1: 
echo ($rows[r_state] == 0)?" 正 点 上 班 ":" 迟 到 "; 
break; 
case 0: 
echo ($rows[r_state] == 0)?" 早 退 ":" 正 点 下 班 "; 
break; 
Case 2: 
echo ($rows[r_state] == 0)?" 正 点 加 班 ":" 晚 点 加 班 "; 
break; 
case 3: 
echo ($rows[r_state] == 0)?" 加 班 早退 ":" 加 班 下 班 "; 
break; 
case 4: 
echo ($rows[r_state] == 0)?" 病 假 "事假 "; 
break; 


I 


<!-- 输 出 备注 > 
<td height="25" align="center" valign="middle"><?php echo ($rows[r_remark] != null)?$rows 
[r_remark]: 无 ; ?></td> 


</tr> 
<?php }?> 


DAO 


i 


< 代码 贴 十 

@ onclick="…": 这 里 的 js 脚本 使 用 的 是 第 三 种 方式 ， 在 表单 元 素 或 标签 中 使 用 。 当 单 击 超 链接 时 ， 将 触发 一 个 窗 
口 事件 ， 打 开 一 个 新 的 窗口 。open() 函 数 的 基本 格式 为 open(url,name,feature)。 其 中 url 为 要 打开 的 地 址 ，name 为 打开 窗 
体 的 名 字 ，features 为 窗 体 属 性 特征 ， 如 宽 、 高 等 。 

@ 根据 u_ id， 也 就 是 参数 u_id 的 值 来 判断 是 哪 种 功能 类 型 。 分 3 种 情况 : 上 下 班 登记 、 病 事假 登记 和 加 班 登记 。 

图 判断 登记 类 型 ， 一 共 分 6 种 情况 : 上班、 下班、 病假、 事假 、 加 班 签到 和 加 班 签 退 。 

@ 判断 状态 类 型 ， 一 共 分 10 种 情况 : 上 班 迟 到 、 上 班 早退 、 正 点 上 班 、 正 点 下 班 、 正 点 加 班 、 加 班 下 班 、 加 班 晚 
点 、 加 班 早 退 、 病 假 和 事假 。 


当 单 击 “ 上 下 班 登记 ” 超 链接 时 ， 就 会 弹出 登记 页 面 ， 用 户 可 以 进行 上 下 班 的 登记 操作 ， 在 “ 登 


记 类 型 ”下 拉 列 表 框 
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原因 ， 最 后 单 击 “ 登 记 ” 按 钮 进行 登记 ， 如 果 登 记 成 功 ， 将 显示 提示 对 话 框 。 登 记 页 面 的 运行 结果 如 


图 3.25 所 示 。 
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图 3.25 登记 页 面 的 运行 结果 
在 登记 界面 中 ， 主 要 有 3 个 表单 元 素 和 一 个 隐藏 表单 ， 如 表 3.8 所 示 。 
表 3.8 上 下 班 登记 的 表单 元 素 


重要 属性 


st" action="p_login_ chk.php" 


元 素 类 型 


form 


名 称 


Pl method=" 


含义 
登记 表单 


text 


u name 


value="<?php echo$ SESSION[u name]: 


登记 人 


?>" readonly=" 


<?php if($_GETIr id] = 14){7?> 


<?php }else if($_GETIr id] = 15){7?> 
select 


<?php }else if($_GETIr id] = 16){?> 


<option value="1"> 上 班 </option> 
<option value="0"> 下 班 </option> 


<option value="4"> 病 假 </option> 
<option value="5"> 事 假 </option> 


<option value="2"> 加 班 签到 </option> 
<option value="3"> 加 班 签 退 </option> 


php } > 


r_remark textarea name="r_remark" rows="5" 
rid hidden value="<?php echo $ GET[r id]; ?>" 
Submit2 submit value=" 回 复 " onclick="return re_ back0:" 


备注 
功能 id 
“登记 ”按钮 


选择 相应 的 登记 类 型 ， 如 上 班 ， 提 交 表 单 后 ， 系 统 就 会 跳 到 处 理 页 (p_login_chk.php) 处 理 。 在 处 
理 页 中 ， 首 先 判断 登记 类 型 (上班 、 下 班 )， 然 后 根据 登记 类 型 取得 相对 应 的 标准 时 间 ， 通 过 两 个 时 间 
的 对 比 ， 得 到 登记 状态 〈 迟 到 、 早 退 、 正 点 上 下 班 )， 最 后 执行 insert 语句 ， 添 加 新 记录 。 程 序 关 键 代 


人 码 如 下 : 
倒 程 15 ”代码 位 置 ， 光盘 \TM\03\oa\kqgl\p_login_chk.php 


<?php 

session_start(); 

include "../conn/conn.php"; 

include "../inc/chec.php"; 

/*1 代表 迟到 ，0 代表 正点 登记 */ 

/判断 功能 类 型 ， 如 上 下 班 登 记 、 加 班 登 记 */ 

if($r id == "14") or ($r_id == "16")){ 

/判断 登记 类 型 ， 如 上 班 、 下 班 等 */ 
if($u_type == 0) 


/开启 session 支持 
// 包 含 数 据 库 链接 文件 
// 包 含 权限 检查 文件 
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$t_sql = "select * from tb_setup where id = 2"; // 取 得 下 班 标准 时 间 
else if($u_type == 1) 

$t_sql = "select * from tb_setup where id = 1"; // 取 得 上 班 标准 时 间 
else if($u_type == 2) 

$t_sql = "select * from tb_setup where id = 3"; // 取 得 加 班 上 班 时 间 
else if($u_type == 3) 

$t_sql = "select * from tb_setup where id = 4"; // 取 得 加 班 下 班 时 间 


和 
St_rst = mysql_query($t_sql,$conn); 
$t_rows = mysql_fetch_row($t_rst); 
$s_time = $t_rows[2]; 1/ 规定 的 标准 时 间 
© S$now time = date("H:i:s"); /| 当前 的 登记 时 间 
© $sql="inserinto tb_register (r_date,r_time,r_ type,r_state,r_remark,r_id,p_id) 
values(".date("Y-m-d").",".date("H:i:s").",".$_POST[u_type].",".((strtotime($now._time) - strtotime($s_time) > 
0)?1:0).",".$_POSTIr_remark].",".$_POSTI[r _id].",".$_SESSION[id].")"; 
Jelse if($r_id == "15") 
$l_sql = "insert into tb_register (r_date,r_time,r_type,r_state,r_remark,r_id,p_id) 
values(".date("Y-m-d").",".date("H:i:s").",".$_POST[u_type].",”,".$_POSTIr_remark].",".$_POSTIr id].",".$_SE 
SSION[id].")"; 
$l_rst = mysql_query($L_sql,$conn); 
A* 登 记 完成 后 ， 关 闭 窗口 */ 
if($l_rst) 
echo "<script>alert(' 登 记 完成 ');window.close();</script>"; 
else 
echo "<script>alert(' 错 误 ');history.go(-1);</script>"; 


Petit ete 


机 


?> 


< 代码 贴 十 

@ date() 时 间 函 数 可 以 通过 不 同 的 组 合 返回 不 同 的 时 间 样 式 。 例 如 程序 中 的 date("H:is")， 还 可 以 加 上 年 、 月 、 日 、 
星期 等 ， 如 date("Y-m-d H:is 1")。 

@ ((strtotime($now_time) - strtotime($s_time) > 0)21:0): 根据 以 上 内 容 得 到 的 SQL 语句 ， 其 中 strtotime 函数 是 PHP 
的 一 个 时 间 函 数 ， 作 用 是 将 一 个 日 期 时 间 描 述 解 析 为 一 个 unix 时 间 戳 ， 如 strtotime("2007-07-11 13:49:29") 输出 的 结果 
就 是 1184132969， 这 时 对 两 个 时 间 进行 计算 ， 就 可 以 得 到 时 间 的 大 小 关系 了 。 


3.8.4 设置 时 间 的 实现 过 程 


国 考勤 管理 使 用 的 数据 表 : tb_register、tb_setup 
考勤 记录 页 面 的 功能 主要 有 两 部 分 : 一 是 显示 所 有 员工 的 考勤 信息 ; 二 是 设置 标准 时 间 。 显 示 所 
有 员工 的 考勤 信息 的 实现 过 程 和 3.8.3 节 中 的 实现 方式 相 类 似 ， 唯 一 不 同 的 就 是 包括 了 全 体 职员 ， 只 要 


略微 改动 SQL 语句 即 可 。 

下 面 主要 来 看 一 下 设置 标准 时 间 页 面 。 在 “考勤 记录 ”页 ”高 二 证 二 
面 中 单 击 “ 设 置 时 间 ” 超 链接 ， 进 入 设置 时 间 页 面 ， 如 图 3.26 PE wl | 
所 示 。 图 3.26 设置 时 间 页 面 


在 本 页 面 中 ， 从 tb_setup 表 中 读 取 预 定义 的 标准 时 间 ， 有 
4 个 标准 时 间 ， 分 别 是 “上班 签到 ”“ 下 班 签 退 ” “加班 签到 ”和 “加 班 签 退 ”。 程序 关键 代码 如 下 : 


@ 
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倒 程 16 ”代码 位 置 ， 光盘 \TM\03\oa\kqgl\set.time.php 


<?php 
session_start(); // 开 启 session 支持 
include "../inc/chec.php"; // 包 含 权限 检测 文件 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文 件 
yi 
<link href="../css/style.css" rel="stylesheet" /> /引入 css 外 部 链接 文件 
<script src="../js/client_js.js"></script> /引入 js 脚本 文件 


<form name="form1" method="post" action="set time_chk.php"> 
<table border="1" cellspacing="0" cellpadding="0" background="../images/bg.jpg"> 
<tr> 
<td width="150" height="25" align="center" valign="middle"> 上 班 签到 </td> 
<td width="150" height="25" align="center" valign="middle"> 下 班 签 退 </td> 
<td width="150" height="25" align="center" valign="middle"> 加 班 签到 </td> 
<td width="150" height="25" align="center" valign="middle"> 加 班 签 退 </td> 


</tr> 
<?php 
$sqlstr = "select * from tb_setup"; // 提 取 表 中 的 标准 时 间 
$result = mysql_query($sqlstr,$conn); // 返 回 结果 集 
$num = 0; // 设 置 序号 
?> 
<tr> 


<?php while($rows = mysql_fetch_row(S$result)X{ ?> 
<td height="30" align="center valign="middle"> 
© <inputtype="text" name="| time<?php echo $num;?>" value="<?php echo $rows[2]; ?>" size=15></td> 
<?php $num++;}?> 
</tr> <tr> 
四 <tdheight="30" align="center" valign="middle"><input type="submit" name="u_logo" value=" 设 置 "></td> 
<td height="30" align="center" valign="middle"><input type="submit" name="d_logo" value=" 设 置 "></td> 
<td height="30" align="center" valign="middle"><input type="submit" name="a_logo" value=" 设 置 "></td> 
<td height="30" align="center" valign="middle"><input type="submit" name="q_logo" value=" 设 置 "></td> 
</tr> 
</table> 
</form> 


< 代码 贴 十 


@ 变量 Snum 是 为 了 在 while 循环 语句 中 输出 不 同 的 表单 元 素 的 名 称 ， 每 次 循环 Snum 自 加 1。 
@ 生成 4 个 设置 时 间 的 按钮 ， 在 处 理 页 中 ， 根 据 不 同 的 名 字 做 不 同 的 处 理 。 


当 调 整 了 其 中 的 一 项 后 ， 单 击 “ 设 置 ”按钮 ， 进 入 到 处 理 页， 在 处 理 页 中 ， 会 根据 按钮 的 名 称 和 


text 元 素 的 value 值 来 进行 相关 的 操作 。 程 序 关键 代码 如 下 : 


倒 程 17 ”代码 位 置 ， 光盘 \TM\03\oa\kqgl\set.time_chk.php 


<?php 
session_start(); // 开 启 session 支持 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../conn/conn.php"; // 包 含 数据 库 链接 文件 


六 根据 名 称 判断 事件 是 哪个 按钮 产生 的 ， 并 生成 不 同 的 SQL 语句 */ 
iflisset($_POST[u logo])){f 
$sqlstr = "update tb_setup set| time =".$_POSTILtime0]." where id = 1"; /设置 上 班 时 间 


_ 国 ) 
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Jelse iflisset($_POST[d logo])i{ 
$sqlstr = "update tb_setup set | time =".$_POSTIL time1]." where id = 2"; // 设 置 下 班 时 间 


jelse if(isset($_POST[a_logo]){ 
$sqlstr = "update tb_setup set |_time =".$_POSTI| time2]." where id = 3"; ”// 设 置 加 班 上 班 时 间 


}else if(isset($_POSTI[q_logo])){ 
$sqlstr = "update tb_setup set |_time =".$_POSTI| time3]." where id = 4"; // 设 置 加 班 下 班 时 间 


} 
else 
echo "<script>alert(' 非 法 登录 ');window.close();</script>"; 


人 人 家 


$result = mysql_query($sqlstr$conn); 
if($result) 
echo "<script>alert(' 设 置 成 功 ');window.close();</script>"; 
else 
echo "<script>alert(' 设 置 失败 ');window.close();</script>"; 
?> 


3.9 后台 首页 设计 


由 于 后 台 管 理 主要 是 针对 管理 员 开放 的 ， 普 通用 户 无 法 访问 到 这 里 ， 所 以 在 页 面 风格 的 设计 上 相对 
要 简单 得 多 。 不 需要 艳丽 的 图 片 、 极 炫 的 特效 ， 只 要 干净 、 整 洁 ， 给 人 一 种 朴实 无 华 的 感觉 是 最 好 的 。 


3.9.1 后 台 首 页 概述 


如 果 说 前 台 是 系统 的 外 衣 ， 那 么 后 台 就 是 系统 的 灵 瑰 了 。 数 据 更 新 、 系 统 升 级 ， 甚 至 是 安全 隐患 ， 
都 源 于 后 台 的 设计 是 否 完善 ， 功 能 是 否 强大 。 可 以 说 ， 后 台 越 完美 ， 系 统 拥 有 的 外 衣 才 会 越 华丽 ， 人 
们 的 目光 才 会 被 吸引 。 

办 公 自 动 化 管理 系统 的 后 台 首页 主要 包括 的 功能 如 下 。 

网 站 首部 导航 栏 : 包括 当前 时 间 、 首 页 、 重 新 登录 和 退出 登录 。 

网 站 左 侧 导航 栏 : 包括 各 个 管理 模块 及 分 类 。 

部 门 管理 模块 : 主要 用 于 对 部 门 信息 的 操 


作 ， 包 括 添加 、 修 改 、 删 除 和 查看 功能 。 RN 
职员 管理 模块 : 主要 用 于 对 职员 信息 的 统 3 FRI 
一 管理 ， 包 括 添加 、 修 改 、 删 除 和 查找 
职员 。 
权限 管理 模块 : 主要 用 于 对 部 门 、 职 员 
用 户 组 的 功能 权限 分 配 。 
系统 管理 模块 : 维护 系统 日 志 、 修改 管理 
员 密 码 和 系统 数据 的 备份 。 


本 案例 中 提供 的 后 台 首 页 如 图 3.27 所 示 。 该 
首页 在 本 书 光盘 中 的 路 径 为 TMVO3\vadminvadmin 
main.php。 


全 


3.27 办 公 自 动 化 管理 系统 后 台 首页 
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3.9.2 ”后台 首 页 技术 分 析 


后 台 页 面 和 前 台 页 面 的 布局 是 相似 的 ， 都 是 采用 “上 方 固 定 ， 左 侧 嵌 套 ” 的 框架 结构 。 在 上 方 杠 
架 (topFrame) 显示 Banner 和 首部 导航 ， 在 左 侧 框架 〈leftFrame) 中 显示 功能 列表 。 中 间 的 显示 区 
(manFrame) 显示 各 级 窗 体 。 


3.9.3 ”后 台 首页 的 实现 过 程 


国 ” ”后台 首页 使 用 的 数据 表 : tb_controller 
下 面 来 看 一 下 后 台 页 面 中 框架 布局 的 应 用 ， 程 序 代码 如 下 : 
倒 程 18 ”代码 位 置 ， 光盘 \TM\03\oa\admin\admin_main.php 


<! 一 定义 顶部 框架 和 框架 内 显示 的 内 容 --> 
<frameset rows="88,*" cols="™*" frameborder="no" border="1" framespacing="0"> 
<frame src="top.php" name="topFrame" scrolling="No" noresize="noresize" id="topFrame" title="topFrame" /> 
< 上 -定义 嵌 套 的 左 侧 框架 和 框架 内 显示 的 内 容 --> 
<frameset rows="*" cols="216,*" framespacing="0" frameborder="no" border="1"> 
<frame src="left.php" name="leftFrame" scrolling="No" noresize="noresize" id="|leftFrame" title="leftFrame" /> 
<!-- 定 义 主 框架 内 容 --> 
<frame src="main.php" name="mainFrame" id="mainFrame" title="mainFrame" /> 
</frameset> 
</frameset> 
<!- 当 浏览 器 不 支持 框架 时 所 显示 的 内 容 --> 
<noframes><body> 
</body> 
</noframes></html> 


3.10 部门 管理 模块 设计 


一 个 成 熟 、 稳 定 的 企业 一 定 要 有 一 个 合理 的 部 门 结构 ， 包 括 每 个 部 门 需要 做 什么 工作 、 部 门 之 间 
的 协调 关系 和 部 门 之 间 的 上 下 级 关系 等 ， 都 要 明确 下 来 ， 这 样 才能 各 尽 其 职 ， 不 会 出 现 资源 浪费 ， 提 
高 企业 整体 的 效率 。 


3.10.1 部门 管理 模块 概述 
部 门 管理 模块 包括 查看 部 门 和 添加 部 门 两 大 部 分 ， 下 面 分 别 介绍 。 


1. 查看 部 门 


查看 各 个 部 门 名 称 和 部 门 之 间 的 上 下 级 关系 ， 还 可 以 对 部 门 的 信息 和 级 别 进行 修改 ， 如 果 是 最 底 
层 的 部 门 ， 则 可 以 对 其 进行 删除 操作 。 
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2. 添加 部 门 
可 以 添加 新 的 部 门 ， 包 括 部 门 名 称 、 上 级 部 门 和 部 门 备注 。 
部 门 管理 框架 如 图 3.28 所 示 。 
本 
录 界面 
各 
| | 查看 部门 | [Li 
个 也 功 [ 从 站 了 
修改 失败 
删除 部 站 
删除 


图 3.28 ”部 门 管理 操作 流程 图 
3.10.2 ”部 门 管理 模块 技术 分 析 


部 门 管理 中 的 技术 难点 是 : 如 何 实 现 查 看 部 门 模块 中 的 动态 显示 部 门 结构 的 功能 。 这 里 用 
JavaScript+ 递 归 函 数 来 实现 。 下 面 详细 讲解 。 

当 用 户 单 击 “ 查 看 部 门 ” 时 ， 系 统 开始 查找 tb_depart 数据 库 。 首 先 找到 根部 门 的 信息 列表 ， 并 判 
断 第 一 个 根部 门 是 否 有 下 级 部 门 。 如 果 没 有 ， 则 直接 输出 部 门 名 称 、 修 改 和 删除 操作 ， 然 后 查找 下 一 
个 根部 门 ; 如 果 有 下 级 部 门 ， 则 调用 自 定义 函数 list_menu($num,$wid,&$m)。 其 中 : 

$num: 为 上 级 部 门 id 号 。 

$wid: 为 表格 宽度 。 

Sm: 为 表单 元 素 id 变量 值 。 

在 这 个 函数 中 ， 执 行 步骤 如 下 : 

(1) 查找 上 级 部 门 为 $num 的 部 门 列表 。 

(2) 判断 第 一 个 部 门 是 否 有 下 级 部 门 。 

(3) 如 果 没 有 ， 则 输出 部 门 名 称 、 修 改 和 删除 操作 ， 然 后 查找 下 一 个 同 级 部 门 。 

(4) 如 果 有 下 级 部 门 ， 则 再 次 调用 list_menu($num,$wid,&$m) 函 数 。 

(5) 重复 步骤 (1) 一 (4)。 

这 时 ， 相 信 读 者 已 经 看 出 : list menu0 正 是 一 个 递归 函数 。 递 归 函 数 就 是 在 当 满 足 一 定 的 条 件 时， 
函数 自己 调用 自己 ， 经 过 层 层 调用 ， 最 终 达 到 解决 问题 的 目的 。list_menu() 函 数 的 代码 如 下 : 
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全 性 19 ”代码 位 置 ， 光盘 \TM\03\oa\admin\inc\func.php 


<?php 
function list_menu($num,$wid,&$m){ 
人 连接 数据 库 */ 
$conn = mysql_connect("localhost","root","root"); 
mysql_select_db("db_office",$conn); 
mysql_query("set names gb2312"); 
J 
作 查 询 同 级 部 门 */ 
$sqlstr = "select * from tb_depart where up_depart = ".$num; 
Sresult = mysql_query($sqlstr, $conn); 
J 
?> 
<!-- 隐 藏 域 --> 
<trid="OpenMe<?php echo $m; ?>" style="display:none;"> 
<td> <table width="<?php echo $wid; ?>%" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td height="25" align="center><table width="<?php echo ($wid -3); ?>%" border="0" cellspacing="0" 
cellpadding="0"> 


<?php 
/* 循 环 输出 同 级 部 门 */ 
while($rows = mysql_fetch_row($result))f 
人 /* 查 看 下 性 部 门 */ 
$sqlstr1 = "select * from tb_depart where up_depart = ". $rows[0]; 
Sresult1 = mysql_query($sqlstr1,$conn); 
$nu = mysql_num_rows($result1); 
人 当前 部 门 没有 下 属 部 门 时 */ 
if(!$nuX 
$m += 1; 
?> 


© <tronMouseOver="this.style.background="#96F7F4"" onMouseOut="this.style.background=""> 

<td width="100%" align="left">&nbsp;&nbsp;&nbsp;<img src="../images/folder.gif" width="16" height= 
"16" border="0"><?php echo $rows[1] ?>-————————: <a href="edit_depart.php?id=<?php echo $rows[0]; ?>"> 
修改 </a>||<a href="del_depart_chk.php?id=<?php echo $rows[0]; ?>" onClick="return cfm();"> 删 除 </a></td></tr> 
<?php 


} 
/当前 部 门 有 下 属 部 门 时 ” 
else{ 
$m += 1; 
?> 
< 上 -鼠标 划 过 样式 -> 
<tr onMouseOver="this.style.background=#96F7F4" onMouseOut="this.style.background=""> 
@ <td>&nbsp;<a href="Javascript:ShowMe(limg<?php echo $m; ?>,OpenMe<?php echo $m; ?>)"> 
<img src="../Images/jia.gif" border="0" alt=" 展 开 " id="img<?php echo $m; ?>"><?php echo $rows[1];?> 
——-<a href="edit_depart.php?id=<?php echo $rows[0]; ?>"> 修 改 </a>|| 删 除 </td></tr> 


目 list_menu($rows[0],$wid,$m); 


从 


</table></td></tr></table></td></tr> 
<?php}?> 
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< 代码 贴 圭 


@ onMouseOver…onMouseOut: onMouseOver 表示 鼠标 指针 停留 在 当前 元 素 上 时 的 状态 ，onMouseOut 事件 表示 饼 


标 指针 离开 当前 元 素 时 的 状态 。 本 行 用 不 同 的 颜色 表示 两 种 不 同 的 事件 . 
@ 当前 部 门 有 下 属 部 门 时 ， 单 击 图 片 或 文字 链接 ， 触 发 一 个 js 事件 ， 显 示 隐 藏 域 。 
四 调用 自身 函数 。 


在 这 个 模块 中 ,同时 使 用 了 另 一 个 自 定义 函数 isbool($id,$dname), 该 函数 将 $Sdname 参数 和 tb_depart 
表 中 的 部 门 名 称 一 一 比较 ， 如 果 重 复 ， 则 返回 tue， 和 否则 返回 false。$id 是 在 修改 部 门 时 ， 先 去 掉 要 修 
改 的 部 门 的 名 称 再 进行 比较 。 在 信息 处 理 页 中 , 根据 返回 的 值 进行 操作 。isbool0 函 数 的 程序 代码 如 下 : 
倒 程 20 ”代码 位 置 ， 光盘 \TM\03\oa\admin\inc\func.php 


function isbool($id,$dname){ 
人 连接 数据 库 */ 
$conn = mysql_connect("localhost", "root","root"); 
mysql_select_db("db_office",$conn); 
mysql_query("set names gb2312"); 


Pr 
$sqlstr = "select * from tb_depart where id != $id d_name = '$dname";  // 创 建 查询 SQL 语句 
$result = mysql_query($sqlstr,$conn); /| 执行 SQL 语句 
if(mysql_num_rows(S$result)>0) /| 判断 是 否 有 记录 集 

$isbool = true; /如 果 有 ，$isbool = true 
else 

$isbool = false; /否则 $isbool = false 
return $isbool; // 返 回 $isbool 


3.10.3 部门 查 看 的 实现 过 程 


国 部门 查 看 模块 使 用 的 数据 表 : tb_depart 

通过 选择 导航 栏 中 的 “部 门 管理 ”/“ 查 看 部 门 ” = EE 
选项 ， 进 入 部 门 查看 页 面 。 在 部 门 查 看 页 面 中 显示 
部 门 设置 及 架构 情况 ， 如 图 3.29 所 示 。 

在 部 门 查看 页 面 中 ， 使 用 到 了 递归 函数 。 首 先 
查找 根部 门 〈 根 部 门 的 条 件 是 top_depart = 0)， 并 
返回 根部 门 的 结果 集 ; 接着 使 用 while 循环 输出 所 
有 根部 门 ， 每 输出 一 个 根部 门 时 ， 都 要 查看 当前 部 
门 是 否 有 子 部 门 〈 下 属 部 门 )。 如 果 没 有 子 部 门 ， 图 3.29 部 门 查看 页 面 
那么 直接 输出 部 门 名 称 和 对 部 门 的 操作 (修改 和 删 
除 ); 如 果 根 部 门 有 子 部 门 ， 则 调用 递归 函数 (ist_menu())。 该 页 面 的 程序 代码 如 下 : 


全 性 21 代码 位 置 : 光盘 \TM\03\oa\admin\bmgl\show_depart.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权 限 检查 文件 


@ 
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include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
include "../inc/func.php"; // 包 含 自 定义 函数 文件 
> 
<script src="../js/admin_js.js"></script> /JI 引 入 js 脚本 文件 
<link href="../css/style.css" rel="stylesheet"> 11 引 入 外 部 css 样式 
<body onLoad="ShowMe(img1,0penMe1)"> /lonLoad 事件 ， 表 示 当 页 面 被 载 入 时 调用 ShowMe() 方 法 


<!-- 查 看 部 门 信息 -> 
<table width="100%" height="25” border="0" cellpadding="0" cellspacing="0"> 
<?php 
人 * 找 到 根部 门 */ 
$sqlstr = "select * from tb_depart where top_depart = 0"; 
$result = mysql_query($sqlstr$conn); 
/隐藏 域 id 号 */ 
$m = 1; 
循环 输出 所 有 根部 门 */ 
while($rows = mysql_fetch_row( $result)X{ 
S$wid = 100; 
* 查 看 下 属 部 门 */ 
S$sqlstr1 = "select * from tb_depart where up_depart = ". $rows[0]; 
$result1 = mysql_query($sqlstr1,$conn); 
Snu = mysql_num_rows($result1); 
人 * 当 前 部 门 没有 下 属 部 门 时 */ 
if(ISnuX{ 
?> 
<tr onMouseOver="this.style.background=#96F7F4" onMouseOut="this.style.background=""> 
<td>&nbsp;&nbsp;&nbsp;&nbsp;<a href="<?php echo $PHP_SELF; ?>"><img src=".JImages/ foldergif" 
border="0”alt=" 展 开 " id="img<?php echo $m; ?>"><?php echo $rows[1]; ?></a>--- 一 一 -一 <a href="edit_ 
depart.php?id=<?php echo $rows[0]; ?>"> 修 改 </a>||<a href="del_depart_chk.php?id=<?php echo $rows[0] ?>" 
onClick="return cfm();"> 删 除 </a></td> 
</tr> 
<?php 
八 当 前 部 门 有 下 属 部 门 时 */ 
}else{ 
?> 
<tr onNMouseOver="this. style.background=#96F7F4"" onMouseOut="this.style.background=""> 
<td><a href="Javascript:ShowMe(img<?php echo $m; ?>,OpenMe<?php echo $m; ?>)"><img src= 
"..JImages/jia.gif' border="0" alt=" 展 开 " id="img<?php echo $m; ?>"><?php echo $rows[1]; ?></a>-— 一 -一 <a 
href="edit_depart.php?id=<?php echo $rows[0]; ?>"> 修 改 </a>|| 删 除 </td> 
</tr> 
<?php 
/输出 同 级 部 门 ， 调 用 递归 函数 小 
list_menu($rows[0],$wid,$m); 
$m += 1; 


人 
/循环 结束 4 
> 
1. 修改 部 门 
在 查看 部 门 页 面 中 ， 可 以 对 部 门 的 名 称 、 上 级 部 门 及 主要 职责 等 相关 信息 进行 修改 。 单 击 要 修改 
部 门 名 称 后 面 的 “修改 ” 超 链 接 ， 进 入 修改 部 门 页 面 ， 如 图 3.30 所 示 。 
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3.30 修改 部 门 页 面 
修改 部 门 页 面 (edit_departphp) 中 的 主要 表单 元 素 如 表 3.9 所 示 。 
表 3.9 修改 部 门 页 面 中 的 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 
d_name Text Size="20" value="<?php echo $rows[d_name] ?> 部 门 名 称 


<?php 
while($tmprows = mysql_fetch_array($tmpresult)){ 
if($tmprows[id] == $rows[top_depart]){ 
vl a echo "<option vale “opeo wd” 列 出 所 有 部 门 
selected>".$tmprows[d_name]."</option>"; 
}else 
echo "<option value=". $tmprows[id].">".$tmprows[d_name]."</option>"; 


}> 


remark, textarea <?php echo $rows[remark] ?> 备注 信息 
id hidden value="<?php echo Srows[id] ?>" 修改 部 门 的 id 
Submit Submit name="Submit" value=" 修 改 " onclick="return a_check();" “修改 ”按钮 


当 管 理 员 修 改 完 相 应 的 信息 后 单 击 “ 修 改 ” 按 钮 提交 修改 信息 ， 系 统 转 到 处 理 页 (edit_depart_ 
chk.php)， 首 先 判 断 输 入 的 部 门 名 称 是 否 重 复 。 如 果 重 复 ， 则 返回 : 如 果 不 重复 ， 则 继续 执行 ， 确 定 修 
改 后 的 上 级 部 门 和 根部 门 。 关 键 代码 如 下 : 

倒 程 22 ”代码 位 置 ， 光盘 \TM\03\oa\ladmin\bmgl\edit_depart_chk.php 


<? 
session_start(); /开启 session 支持 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
include "../inc/func.php"; // 包 含 数据 库 链接 文件 
/判断 输入 部 门 名 称 是 否 重复 */ 


if(isbool($_POST[id],$_POST[Id_name])X 
echo "<script>alert( 名 称 已 存在 ， 请 重新 输入 山 );history.go(-1);</script>"; 
exit(); 


/修改 部 门 ， 确 定 上 级 部 门 和 根部 门 ?/ 
if($_POST[u id] != 0X 
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/* 查 找 上 级 部 门 的 根部 门 */ 
$sqlstr = "select top_depart from tb_depart where id = ".$_POST[u_id]; 
Sresult = mysql_query($sqlstr, $conn); 
$rows = mysql_fetch_array($result); 
/* 如 果 存 在 上 级 部 门 ， 就 取得 上 级 部 门 的 根部 门 */ 
if ($rows[top_depart] != 0) 
S$top_depart = $rows[top_depat]; 
/* 如 果 不 存在 ， 将 上 级 部 门 作为 根部 门 */ 
else 
Stop_depart =$_POST[u_id]; 
Jelse 
"如果 没有 上 级 部 门 ， 那 么 就 将 自身 定 为 根部 门 */ 
S$top_depart = 0; 
“创建 、 执 行 修改 部 门 的 SQL 语句 */ 
S$sqlstr = "update tb_depart set d_name = ".$_POSTId_name].",top_depart = ".$top_depart.", up_depart = 
"$_POST[u id].", remark = ".$_POST[remark]." where id = "$_POST[id]; 
Sresult = mysql_query($sqlstr, $conn); 
re_message($result,"show_depart.php"); 
?> 


2. 删除 部 门 

如 果 要 取消 某 个 部 门 ， 可 以 将 该 部 门 删除 。 在 删除 部 门 时 ， 只 能 对 最 底层 的 部 门 进行 删除 ， 如 果 
部 门下 面 有 子 部 门 ， 其 部 门 后 面 的 “删除 ” 超 链接 不 可 用 。 删 除 功能 的 实现 相对 比较 简单 一 点 ， 单 击 
“删除 ” 超 链接 后 ， 系 统 提示 “是 否 确认 删除 部 门 ” 如 果 单 击 “ 确 认 ” 按 钮 ， 系 统 将 转 到 处 理 页 (del_ 
depart_chk.php) 进行 删除 处 理 。 程 序 代 码 如 下 : 

倒 程 23 ”代码 位 置 ， 光盘 \TM\WO3\oavadmin\bmglvdel_depart_chk.php 


<?php 

session_start(); // 开 启 session 支持 
include "../inc/chec.php"; // 包 含 权限 查看 文件 
include "../conn/conn.php"; // 包 含 数据 库 链接 文件 
include "../inc/func.php"; // 包 含 自 定义 文件 
$sqlstr = "delete from tb_depart where id = ".$_GETI[id]; /删除 SQL 语句 
$result = mysql_query($sqlstr,$conn); /| 执行 SQL 语句 
re_message($result,"show_depart.php"); // 返 回 最 后 结果 

?> 


3.10.4 ”部 门 添 加 的 实现 过 程 


用 户 单 击 导航 栏 中 的 “部 门 管理 ”/“ 添 加 部 
门 ”选项 ， 进 入 添加 部 门 页 面 (add_depart.php)， 
该 页 面 用 于 设置 新 添加 部 门 的 相关 信息 , 包括 部 门 
名 称 、 上 级 部 门 和 备注 ， 如 图 3.31 所 示 。 

添加 部 门 页 面 (add_depart) 主要 包含 一 个 添 
加 部 门 表单 ， 表 单元 素 的 名 称 、 说 明 及 属性 值 如 
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表 3.10 所 示 。 
表 3.10 添加 部 门 页 面 的 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含 义 
d name | Text | name="d name" id="d_name" size="20" 部 门 名 称 
<?php 
hile(®) = 1_fetch Sresultl 和 
uid select Wh ee oC ea) 列 出 所 有 部 门 
echo "<option value=".$rows[0].">".$rows[1]."</option>"; 
27> 


="remark" 备注 信息 
“添加 ”按钮 


textarea name="remark" cols="40" rows="5" id 


Submit name="Submit" value=" 添 加 " onclick="retum a_check(;" 


当 管 理 员 提交 表单 后 ， 进 入 处 理 页 (add_depart_chk.php) 进行 添加 操作 。 在 处 理 页 中 ， 首 先 判 断 
输入 的 部 门 名 称 是 否 重复 ,如 果 名 称 符合 要 求 ， 则 进一步 确定 上 级 部 门 和 根部 门 ， 最 后 执行 insert 语句 
进行 添加 操作 。 处 理 页 的 代码 如 下 : 

倒 各 24 ”代码 位 置 ， 光盘 \TM\03\oa\admin\bmgl\add_depart_chk.php 


<? 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../conn/conn.php"; /包含 数据 库 链 接 文件 
include "../inc/func.php"; /包含 自 定义 函数 文件 
/判断 输入 部 门 名 称 是 否 重复 "/ 


ifisbool($_POST[d_name])i{f 
echo "<script>alert(' 名 称 已 存在 ， 请 重新 输入 山 );history.go(-1);</script>"; 
exit(); 


} 
/确定 上 级 部 门 和 根部 门 六 
if$_POST[u_id] (= "O"){ 
$sqlstr = "select top_depart from tb_depart where id = ".-$_POST[u_id]; 
$result = mysql_query($sqlstr,$conn); 
$rows = mysql_fetch_row($result); 
if ($rows[top_depart] != 0) 
S$top_depart = $rows[top_depart]; 
else 
S$top_depart =$_POST[u_id]; 
} 
else 
S$top_depart = 0; 
人 添加 新 部 门 */ 
$sqlstr = "insert into tb_depart 
values(",".$_POSTI[d_name].",".$top_depart.",".$_POSTI[u_id].",".$_POST[remark].")"; 
$result = mysql_query($sqlstr$conn); 
/输出 添加 结果 ”/ 
re_message($result,"show_depart.php") 
?> 


全 
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3.10.5 ”单元 测试 


本 模块 使 用 了 JavaScript 和 自 定义 函数 来 显示 部 门 结构 列表 , 在 进行 测试 时 发 现 ， 当 部 门 被 删除 到 
只 有 一 个 根部 门 时 ， 就 会 弹出 错误 提示 窗口 ， 如 图 3.32 所 示 。 


A 


是 否 怒 续 在 该 页 面 上 运行 央 本 程序 ? 


图 3.32 出错 信息 
单 击 “ 是 ”按钮 ， 其 他 操作 一 切 正常 。 经 检查 发 现 ， 在 当前 页 面 被 载 入 时 调用 了 js 脚本 ,语句 如 下 : 


<body onLoad="ShowMe(img1,OpenMe1)"> 


在 部 门 被 删 到 只 有 一 个 时 ， 变 量 OpenMel 就 不 再 存在 ， 调 用 函数 ShowMe() 时 就 会 报错 ， 这 时 市 
除 “onLoad="ShowMe(img1,OpenMe1)"”， 错 误 就 不 会 再 出 现 。 


3.11 系统 管理 模块 设计 


系统 管理 模块 在 后 台 系统 中 占据 着 非常 重要 的 位 置 , 后 台 其 他 的 模块 都 是 针对 前 台 功 能 而 设计 的 。 
而 系统 管理 是 针对 系统 本 身 的 一 些 信息 、 情 况 而 设计 的 ， 一 般 包括 数据 备份 、 管 理 员 信 息 修改 等 ， 甚 
至 可 以 对 操作 系统 的 配置 文件 进行 操作 ， 可 以 说 是 管理 员 的 管理 员 ， 是 后 台 系 统 中 必 不 可 少 的 组 成 


3.11.1 系统 管理 模块 概述 


系统 管理 模块 包括 如 下 功能 。 


系统 日 志 ， 记 录 了 前 、 后 台 用 户 登 录 时 的 _ 
相关 信息 。 四 

数据 备份 : 可 以 对 整个 数据 库 进行 备份 、 一 > 网 
恢复 和 删除 。 a = 

修改 密码 :修改 后 台 管 理 员 登 录 密 码 。 

系统 管理 用 例 图 如 图 3.33 所 示 。 图 3.33 系统 管理 用 例 图 


3.11.2 ”系统 管理 模块 技术 分 析 
在 系统 管理 模块 中 ， 保 存 数据 的 方式 使 用 了 数据 库 和 文件 ， 关 于 数据 库 技术 已 经 介绍 很 多 了 ， 本 
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节 主要 讲解 一 下 文件 及 文件 夹 的 相关 操作 。 
对 文件 的 操作 主要 分 3 步 : 打开 文件 ， 读 取 / 写 入 文件 和 关闭 文件 。 
1. 打开 文件 
打开 文件 使 用 的 是 fopen0 函 数 ， 该 函数 的 一 般 格 式 如 下 : 
int fopen (string filename,string mode ); 


参数 filename 是 要 打开 的 包含 路 径 的 文件 名 ， 可 以 是 相对 路 径 或 绝对 路 径 。 如 果 没 有 任何 前 级 ， 
则 表示 打开 的 是 本 地 文件 ; 参数 mode 是 打开 文件 的 方式 , 常用 的 有 a (追加 )、w (只 写 ) 和 r (只 读 )。 
代码 如 下 : 


$f_open = fopen("../bak/remark.sql","a+"); 


2. 读 取 / 写 入 文件 


读 取 文件 的 函数 有 很 多 ， 最 常用 的 就 是 按 行 读 取 (fgets0 函 数 ) 和 全 部 读 取 (readfile() 函 数 和 
fpassthru0) 函 数 )。 
fgets() 函 数 的 一 般 格式 如 下 : 


string fgets ( int filename [ ,int length ]) ; 


参数 filename 是 被 打开 的 文件 ， 参 数 length 是 要 读 取 的 数据 长 度 。 该 函数 能 够 实现 从 filename 指 
定 的 文件 中 读 取 一 行 并 返回 长 度 最 大 值 为 length-1 个 字 节 的 字符 串 。 在 遇 到 换行 符 、EOF 或 者 读 取 了 
length-1 个 字 节 后 停止 。 如 果 没 有 指定 length 的 长 度 ， 默 认 值 是 IKB。 代 码 如 下 : 
<?php 
$f_open = fopen ("../bak/remak.txt","r") ; /打开 文本 
echo fgets ( $f_open) ; // 读 取 其 中 的 数据 


fclose ( $f_open ) ; 
?> 


readfile() 函 数 的 一 般 格式 如 下 : 


int readfile ($filename ); 


参数 $filename 为 要 打开 的 文件 名 ， 使 用 该 函数 时 ， 无 须 进行 第 1、3 步 ， 也 无 须 输出 语句 echo， 
直接 就 可 以 读 取 文 件 的 全 部 内 容 并 显示 在 页 面 中 。 


3. 关闭 文件 

完成 对 文件 的 操作 后 ， 就 可 以 关闭 文件 了 。 关 闭 文件 的 命令 十 分 简单 ， 格 式 如 下 : 

fclose ( $open); 

其 中 参数 $open 是 要 关闭 的 文件 资源 。 

在 本 模块 中 ， 还 用 到 了 两 个 自 定 义 函 数 c_ log0 和 show_file0。c_logO0 函 数 的 作用 是 删除 日 志文 件 ， 
程序 代码 如 下 : 


@ 
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倒 程 25 ”代码 位 置 ， 光盘 \TM\03\oa\ladmin\inc\func.php 


function c_log(X{ 
Sfilename="../log.txt"; 
if(file_exists($filename)) 
unlink($filename); 
else 


// 日 志文 件 路 径 

// 判 断 文件 是 否 存在 

// 如 果 文件 存在 ， 删 除 

// 如 果 文件 不 存在 ， 返 回 上 层 操作 


echo "<script>alert(' 暂 无 系统 日 志 ! ');history.go(-1);</script>"; 


} 


show_file() 函 数 的 作用 是 返回 一 个 目录 下 的 所 有 文件 名 列表 ， 并 以 数组 的 形式 返回 。show_file() 函 


数 的 程序 代码 如 下 : 


倒 程 26 ”代码 位 置 ， 光盘 \TM\03\oa\ladmin\inc\func.php 


function show file(X 

S$folder_name = "../bak"; 

$d_open = opendir($folder_name); 

$num = 0; 

while($file = readdir($d_open)){ 
S$filename[$num] = $file; 
$num++; 

: 

closedir($d_open); 

return $filename; 


/| 数据 备份 目录 

// 打 开 备 份 目 录 
/数组 下 标 

/| 循环 读 取 文件 

// 将 文件 名 存 为 数组 


// 关 闭 目录 
// 返 回 文件 数组 


3.11.3 ”系统 日 志 的 实现 过 程 


通过 导航 栏 中 的 “系统 管理 ”/“ 系 统 日 志 ” 可 进入 系统 
日 志 页 面 。 在 该 页 面 中 显示 系统 日 志 信息 ， 在 系统 日 志 中 ,， 记 
录 着 普通 用 户 和 后 台 管理 员 所 有 人 员 登 录 系 统 时 的 信息 , 包括 
登录 时 间 、 用 户 账号 、 登 录 IP 等 。 这 些 信 息 是 在 用 户 登 录 时 
系统 自动 添加 到 数据 表 中 的 (关于 用 户 登录 的 实现 请 参见 本 书 
附带 的 光盘 )。 系 统 日 志 页 面 的 运行 结果 如 图 3.34 所 示 。 

显示 日 志 是 对 文件 进行 读 取 ， 定 位 到 日 志文 件 log.txt 后 ， 


图 3.34 系统 日 志 页 面 的 运行 结果 


使 用 fgets() 函 数 读 取 日 志 内 容 ， 并 显示 在 页 面 中 。 程 序 代码 如 下 : 


倒 程 27 ”代码 位 置 ， 光盘 \TM\03\oa\admin\xtgl\slog.php 


<?php 
session_start(); 
include "../inc/chec.php"; 
?> 
<script src="../js/admin_js.js"></script> 
<link href="../css/style.css" rel="stylesheet"> 


<tr> 


/开启 session 支持 
// 包 含 权限 检查 文件 


/1 引入 js 脚本 文件 
/引入 外 部 css 样式 文件 


<td colspan="4" height="30" align="center" valign="middle"><a href="del slog_chk.php" onclick= 


"return del_chk();"> 清 除 日 志 </a></td> 


匠 
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</tr> 
<?php 
$filename = "../log.txt"; /日 志文 件 路 径 
if($f_open = fopen($filename,"r")) // 是 否 以 只 读 形式 打开 文件 
while($str = fgets($f_open,255)X{ / 按 行 读 取 日 志文 件 内 容 
$chr = split(",",$str); // 以 “, ”为 分 隔 符 ， 返 回 数组 形式 的 记录 集 
echo "<tr>"; 
for($i = 0; $i < count($chr); $i++X{ /| 循环 输出 字段 值 
echo "<td align='center height='25'>".$chr[$i]."</td>"; 
echo "</tr>";} 
fclose($f_open); /关闭 文件 
} else // 如 果 没 有 日 志文 件 时 
echo "<script>alert( 还 没有 日 志文 件 ! ');history.go(-1);</script>"; 
?> 
</table> 
</td></tr></table> 


对 于 无 用 的 日 志 信息 需要 定期 清除 ， 通 过 页 面 中 的 “清除 日 志 ” 超 链接 清空 日 志 ， 实 际 上 就 是 删 
除 日 志文 件 。 程 序 代码 如 下 : 
倒 程 28 ”代码 位 置 ; 光盘 \TMW3\oavadmin\xtgldel_slog_chk.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../inc/func.php"; /包含 自 定义 函数 文件 
c_log(); // 调 用 自 定 义 函 数 


echo "<script>alert(' 删 除 成 功 ! ');location='data_stock.php';</script>"; 


3.11.4 数据 备份 的 实现 过 程 


在 病毒 肆虐 的 今天 ， 数 据 备份 的 重要 性 是 毋 需 置疑 的 。 2 ee 
没有 任何 一 家 网 站 、 一 个 系统 敢 说 自己 是 绝对 安全 的 , 只 要 Eee | ia 
稍 不 留神 就 会 中 招 。 本 着 对 读者 负责 的 态度 ， 在 3.13 节 的 。 cl 

技术 专题 中 ， 将 详细 地 介绍 MySQL 数据 的 备份 与 恢复 ， 以 。“” 记 s 
及 如 何在 PHP 中 对 MySQL 进行 备份 操作 。 这 里 先 简单 了 解 。 a 
一 下 本 系统 数据 备份 的 实现 过 程 。 数据 备份 页 面 的 运行 结果 。 


如 图 3.35 所 示 。 se 
当 单 击 “ 备 份 数据 ”按钮 时 ， 系 统 执行 exec0 函 数 ， 该 图 3.35 数据 备份 页 面 的 运行 结果 
函数 的 作用 是 执行 系统 命令 ， 通 过 系统 命令 来 备份 数据 。 数 
据 备份 的 程序 代码 如 下 : 
倒 程 29 ”代码 位 置 ， 光盘 \TM\03\oavadminwtglvdata_stock_chk.php 
<?php 
session_start(); /开启 session 支持 


@ 
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include "../inc/chec.php"; // 包 含 权限 检查 文件 

$mysqlstr = "D:\AppSem\MySQL\\bin\mysqldump -uroot -h192.168.1.149 -proot -opt -B db_office 
> .bak/".$_POST[b_name]; /数据库 备份 语句 

exec($mysqlstr); /| 执行 系统 命令 


echo "<script>alert(' 备 份 成 功 ');location='data_stock.php'</script>"; 
> 
恢复 数据 和 备份 数据 的 执行 过 程 大 同 小 异 , 都 需要 使 用 exec() 函 数 执行 系统 命令 , 只 是 命令 执行 的 
操作 是 完全 相反 的 。 恢 复数 据 的 程序 代码 如 下 : 
倒 程 30 ”代码 位 置 ， 光盘 \TM\03\oa\admin\xtgl\rebak_stock_chk.php 
<?php 


session_start(); /开启 session 支持 

include "../inc/chec.php"; /包含 权限 检查 文件 

$mysqlstr = "D:\AppSerW\\IMySQL\\bin\mysql -uroot -h192.168.1.149 -proot db_office < ../bak/".$_POST 
[r_name]; /数据 恢复 命令 

exec($mysqlstr); /执行 系统 命令 


echo "<script>alert(' 恢 复 成 功 ');location='data_stock.php'</script>"; 
2 
删除 数据 备份 实质 就 是 删除 备份 文件 ， 先 使 用 自 定义 函数 show_file0 返 回 一 个 文件 名 列表 数组 ， 
再 使 用 for 循环 语句 ， 逐 一 删除 备份 文件 。 程 序 代 码 如 下 : 
倒 程 31 ”代码 位 置 ， 光盘 \TM\03\oa\ladmin\xtgl\del_stock_chk.php 


<?php 
session_start(); /开启 session 支持 
include "../inc/chec.php"; /包含 权限 检查 文件 
include "../inc/func.php"; /包含 自 定义 函数 文件 
$filename = show_file(); ll 调用 自 定义 函数 show_file()， 返 回 文件 名 列表 


for($num = 2;$num < count($filename);$num++){ ”// 循 环 删除 文件 
unlink("../bak/".$filename[$num]); 


} 
echo "<script>alert(' 删 除 成 功 ! ');location='data_stock.php'</script>"; 
?> 


3.12 ”开发 技巧 与 难点 分 析 


3.12.1 使 用 JavaScript 关联 多 选 列 表 框 


在 本 系统 中 ， 多 个 模块 都 使 用 了 这 种 技术 ， 即 在 两 个 多 选 列表 框 之 间 产 生 关联 ， 从 列表 框 A 中 删 
除 的 数据 被 相应 地 添加 到 列表 框 B 中 ， 反 之 从 列表 框 B 中 删除 的 数据 会 被 添加 到 列表 框 A 中 。 
在 权限 管理 模块 中 的 用 户 组 设置 ， 就 是 用 这 种 技术 实现 的 “添加 新 用 户 组 ”功能 ， 首 先 在 左 侧 的 
多 选 列 表 框 中 选取 要 添加 的 成 员 ， 单 击 “ 添 加 组 员 ” 超 链接 ， 被 选中 的 成 员 就 被 添加 到 右 侧 列表 框 ， 
同时 删除 在 左 侧 列表 框 中 的 成 员 ， 运 行 结果 如 图 3.36 所 示 。 
国 
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图 3.36 使 用 JavaScript 关联 多 选 列表 框 


下 面 来 看 一 下 具体 的 操作 ， 首 先 创建 一 个 表单 ， 其 中 两 个 多 选 列表 框 为 必 选 的 表单 元 素 ， 分 别 命 
名 为 left 和 right， 放 在 左 、 右 两 边 ， 其 中 left 列表 框 中 的 数据 是 从 数据 库 中 读 取 的 ，right 列表 框 则 初 
始 为 室 。 程 序 代码 如 下 : 


倒 程 32 代码 位 置 ， 光盘 \TM\03\oa\admin\qxgl\add_group.php 


<!-- 输 出 左 侧 列表 框 -> 
<SELECT name="left" size="10" multiple style="width:100px; "> 
<?php 
while($rows = mysql_fetch_row(S$result)){ 
echo "<option value=". $rows[1].">".$rows[1]."</option>"; /| 从 数据 库 中 读 取 列 表 值 
} 
?> 
</SELECT> 
<!- 2 - -> 
</td> 
<!-“ 添 加 ”“ 删 除 ” 超 链接 --> 
<td width="96" align="center" valign="middle"> 
<a href="#" onClick="activeList(document.form1.left,document.form1.right)"> 添 加 组 员 &gt;&gt;</a><br> 
<br> 
<a href="#" onClick="activeList(document.form1.right,document.form1.left)">&lt;&lt; 删 除 组 员 </a></td> 
<!-- 右 侧 列表 框 --> 
<td colspan="2" align="center valign="middle"><select name="right" size="10" multiple style="width:100px; "> 
</select></td> 
</tr> 
<tr> 
<td height="30" colspan="4" align="center valign="middle"> 
<input type="hidden" name="g_list" /> 
<!--“ 添 加 ”按钮 -> 
<input type="submit" value=" 添 加 " onclick="return glist()" /><input type="reset" value=" 重 置 " /> 


上 面 的 代码 主要 是 HTML 和 PHP 语言 ， 和 普通 的 页 面 没有 什么 区 别 , 核心 的 语句 就 是 代码 中 的 加 
粗 部 分 〈 不 是 注释 )， 当 单 击 “ 添 加 组 员 ” 或 “删除 组 员 ” 超 链接 时 ， 将 触发 JavaScript 脚本 事件 ， 执 
行 activeListO) 函 数 。 函 数 代码 如 下 : 

倒 程 33 ”代码 位 置 ， 光盘 \TM\03\oa\ladmin\js\admin_jsjs 


/关联 多 选 列表 
*headStream: 源 多 选 列表 
*endStream: 目的 多 选 列表 


@ 
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了 
function activeList(headStream,endStream}{ 
var valueList = new Array(); /| 数组 ， 用 来 存储 要 移动 数据 的 value 值 
var textList = new Array(); /| 数组 ， 用 来 存储 要 移动 数据 的 text 值 
var valueTmpText = new Array(); // 数 组， 用 来 存储 要 移动 数据 的 option 项 
varindex = 0; /| 数组 下 标 
* 存 储 源 列表 中 被 选中 的 数据 */ 
for(var i=0; i<headStream.options.length; i++ 并 
if(headStream.options[il.selected){ // 判 断 元 素 是 否 被 选中 
valueList[index] = headStream.options[i].value; /存储 value 值 
textList[index] = headStream.options[i] .text; /存储 text 值 
valueTmpText[valueList[index]] = headStream.options[j]; /存储 option 项 
index ++; 


» 
/向 目的 列表 中 添加 数据 ， 同 时 删除 对 应 的 源 列 表 数 据 ?/ 

for(var i=0; i<textList.length; i++){ 
var foption = document.createElement("option"); /建立 新 的 option 项 
foption .text = textList[j]; 
foption.value = valueList[j]; 
endStream.add(foption); // 向 目标 列表 添加 新 建 option 项 
headStream.removeChild(valueTmpTextfvalueList[]]); /从 源 列表 中 移 除 option 项 

及 


功 卫 分 光 喇 本 区 


3.12.2 ”用户 组 设置 


日、 全 山谷 息 


在 后 台 的 权限 管理 模块 中 ， 放 弃 了 常用 的 “ 读 ”、“ 写 ”这 样 的 党 i 
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规 操 作 ， 而 是 采用 了 用 户 组 的 模式 。 用 户 组 的 设置 主要 是 两 方面 : 一 
方面 是 用 户 组 成 员 ， 同 一 用 户 可 以 分 配 在 不 同 的 用 户 组 ; 另 一 方面 是 
功能 列表 ， 可 以 设置 用 户 组 的 访问 权限 ， 不 同 的 用 户 组 ， 所 看 到 的 功 
能 列表 也 会 不 同 。 这 样 ， 可 以 使 用 户 的 权利 更 加 细 化 ， 防 止 居心 下 测 | “2m 
的 用 户 恶 意 破 坏 。 用 户 组 A 和 用 户 组 B 的 对 比 结果 如 图 3.37 所 示 。 

在 功能 列表 的 设置 中 ， 首 先 默 认 所 有 的 功能 都 是 可 访问 的 ， 当 有 图 337 用 户 组 权限 对 比 
的 功能 需要 进行 限制 时 ， 就 将 其 从 用 户 组 中 删除 。 程 序 代码 如 下 : 

倒 程 34 ”代码 位 置 ， 光盘 \TM\03\oa\admin\qxgl\pur_assign.php 


<?php 
session_start(); // 开 启 session 支持 
include "../inc/chec.php"; // 包 含 权限 检查 文件 
include "../conn/conn.php"; // 包 含 数据 库 链 接 文件 
$sqlstr = "select id,f_name,o_group from tb_list"; /生成 SQL 语句 

?> 

<script src="../js/admin_js.js"></script> /引入 js 脚本 文件 

<link href="../css/style.css" rel="stylesheet"> /引入 外 部 css 样式 文件 


<tr><td width="75" height="20" align="center" valign="middle"> 功 能 </td> 
<td width="75" height="20" align="center" valign="middle"> 开 放 组 </td> 
<td width="75" height="20" align="center" valign="middle"> 操 作 </td></tr> 
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<?php 

人 输出 功能 列表 项 */ 

Sresult = mysql_query($sqlstr, $conn); 

while($rows = mysql_fetch_row(S$result)X{ 
echo "<tr>"; 
for($i = 1;$i < count($rows); $i++){ 

echo "<td align=center valign=middle>". $rows[S$i]."</td>"; 

echo "<td align=center valign=middle><a href='modify_assign.php?id=".$rows[0]."> 修 改 </a></td>"; 
echo "</tr>"; 


} 


Fg 
5 

</table> 

</td></tr></table> 


本 节 是 3.12.1 节 关 联 多 选 列表 框 的 延伸 ， 是 具体 应 用 关联 列表 实现 的 一 项 功能 。 在 实际 的 应 用 中 ， 
应 多 开动 脑筋 ， 才 能 用 平凡 的 技术 实现 不 平凡 的 功能 。 


3.13 ”MySQL 数据 备份 专题 


关于 备份 数据 的 重要 性 ， 这 里 就 不 再 多 做 介绍 。 本 节 先 来 了 解 常用 数据 库 备份 的 方式 ， 再 学 习 
MySQL 数据 备份 与 恢复 的 方法 ， 最 后 看 一 看 在 PHP 中 是 如 何 对 MySQL 数据 进行 备份 的 。 
1. 数据 库 备份 类 型 


(1) 完全 备份 

这 是 大 多 数 人 常用 的 方式 ， 它 可 以 备份 整个 数据 库 ， 但 需要 花费 更 多 的 时 间 和 空间 ， 所 以 一 般 推 
荐 一 周 做 一 次 。 

(2) 事务 日 志 备份 

事务 日 志 就 是 对 数据 库 操 作 的 记录 ， 它 记录 数据 库 的 改变 ， 备 份 时 只 需要 备份 自 上 次 备份 以 来 对 
数据 库 所 做 的 改变 ， 所 以 只 需要 很 少 的 时 间 。 推 荐 每 小 时 一 次 。 

(3) 差异 备份 

差异 备份 也 叫 增 量 备份 。 它 是 另 一 种 只 备份 一 小 部 分 数据 库 的 方法 ,但 它 不 使 用 事务 日 志 ， 相反 ， 
它 使 用 整个 数据 库 的 一 种 新 映 象 。 差 异 备份 比 最 初 的 完全 备份 小 ， 因 为 它 只 包含 自 上 次 完全 备份 以 来 
所 改变 的 数据 库 。 其 优点 是 存储 和 恢复 速度 快 。 推 荐 每 天 做 一 次 。 

(4) 文件 备份 

数据 库 可 以 由 硬盘 上 的 许多 文件 构成 。 如 果 这 个 数据 库 非常 大 ， 恐 怕 一 个 晚上 也 不 能 将 它 备份 完 ， 
可 以 每 天 备份 数据 库 的 一 部 分 。 由 于 一 般 情况 下 数据 库 不 会 大 到 必须 使 用 多 个 文件 存储 ， 所 以 这 种 备 
份 不 是 很 常用 。 

2. MySQL 数据 备份 与 恢复 


MySQL 中 的 备份 是 非常 灵活 的 , 既 可 以 对 单独 的 数据 表 进 行 备份 , 也 可 以 对 整个 数据 库 进行 备份 ， 
下 面 分 别 进行 介绍 。 
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(1) 数据 表 备份 
备份 数据 表 的 语法 如 下 : 


SELECT * INTO OUTFILE ‘fle_name' FROM tbl_name 


回 file name: 备份 的 文件 名 ， 如 usersql。 

回 tbl_name: 备份 的 数据 表 名 ， 如 tb_users。 

使 用 该 备份 语句 首先 要 登录 到 MySQL， 并 进入 到 相对 应 的 数据 库 中 ， 如 要 备份 的 表 tb_users 是 属 
于 数据 库 db_office 的 ， 备 份 文件 名 为 users.sql， 则 完整 的 备份 流程 如 图 3.38 所 示 。 
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图 3.38 ”数据 表 备份 
备份 后 的 文件 保存 在 数据 表 所 在 目录 中 ， 如 本 例 中 表 tb_users 所 在 的 目录 为 D\AppSerwWMySQL\ 
datavdb offfce， 则 备份 的 sql 文件 就 保存 在 这 个 路 径 下 。 
如 果 要 恢复 备份 文件 ， 代 码 如 下 : 


LOAD DATA INFILE 'user.sql' REPLACE into table ‘tb_users' 


(2) 数据 库 备份 
数据 库 备 份 时 ， 可 以 不 用 登录 到 SQL 中 ， 直 接 使 用 mysqldump 命令 即 可 。 该 命令 格式 如 下 : 


mysqldump -uroot -proot db_office > c:\remark.txt 


备份 的 数据 表 ， 实 际 就 是 一 个 完整 的 建 库 流程 ， 如 果 要 恢复 备份 ， 直 接 用 MySQL 命令 即 可 。 语 
名 如 下 : 


mysql -uroot -proot db_office < c:Wemark.txt 

mysqldump 语句 的 功能 十 分 强大 ， 备 份 只 是 其 中 的 一 个 功能 ， 感 兴趣 的 读者 可 以 参考 mysqldump 
的 帮助 文档 。 

3. 在 PHP 中 备份 数据 库 


在 PHP 中 备份 数据 库 ， 主 要 通过 exec(0) 函 数 执行 系统 命令 。 
exec() 函 数 的 格式 如 下 : 
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string exec ( string command ) 


参数 command 为 要 执行 的 系统 命令 。 这 里 为 备份 数据 表 的 命令 ， 代 码 如 下 : 


exec("D:\\AppSerM\MySQL\\bin\Imysqldump -uroot -h192.168.1.149 -proot -opt -B db_office > ..Jbakremark.sql"); 


《注意 虽然 在 安装 时 MySQL 已 经 设置 了 环境 变量 ， 但 如 果 要 在 PHP 中 执行 mysqldump、mysql 
命令 ， 则 必须 给 出 命令 的 完整 路 径 ; 否则 ， 就 会 出 现 文件 大 小 为 OKB 的 结果 。 


3.14 本 章 总 结 


本 章 通过 一 个 完整 的 办 公 自 动 化 管理 系统 开发 过 程 ， 向 读者 讲述 了 系统 开发 的 几 个 重要 环节 ， 还 
包括 几 个 主要 模块 的 功能 实现 ， 最 后 又 着 重 介绍 了 MySQL 备份 的 相关 知识 。 通 过 本 章 的 学 习 ， 读 者 
对 如 何 开发 一 个 真实 的 项 目 会 有 进一步 的 了 解 。 相 信 读 者 通过 自己 动手 实践 ， 完 成 一 个 功能 相似 的 办 
公 自 动 化 管理 系统 是 完全 不 成 问题 的 。 
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随 着 科技 的 发 展 ， 网 络 技术 已 经 深入 到 人 们 的 日 常生 活 中 ， 同 时 
带 来 了 教育 方式 的 一 次 变革 。 而 网 络 考试 则 是 一 个 很 重要 的 方向 。 基 
于 Web 技术 的 网 络 考试 系统 可 以 借助 于 遍布 全 球 的 Internet 进行 。 因 
此 考试 既 可 以 在 本 地 进行 ， 也 可 以 在 异地 进行 ， 大 大 拓展 了 考试 的 灵 
活性 ， 并 且 缩 短 了 传统 考试 要 求 老师 打印 试卷 、 安 排 者 试 、 监 者 、 收 
集 试 卷 、 评 改 试卷 、 讲 评 试卷 和 分 析 试 卷 这 个 漫长 而 复杂 的 过 程 ， 使 
考试 更 趋 于 容 观 、 公 正 。 本 章 介 绍 了 一 个 具有 在 线 考试 、 即 时 阅卷 、 
成 绩 查 询 以 及 考题 和 考生 信息 管理 等 功能 的 网 络 在 线 考试 系统 。 通 过 
阅读 本 章 ， 可 以 学 习 到 : 

H。 掌 栓 网 络 在 线 考试 的 开发 过 程 

H 掌握 应 用 SQL Server 2000 创建 数据 库 、 数 据 表 的 方法 

掌握 连接 SQL Server 数据 库 的 不 同方 式 

MW 实时 显示 时 间 的 两 种 不 同方 法 

掌握 Ajax 无 刷新 技术 
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4.1 开发 背景 


随 着 教育 改革 的 不 断 深入 ， 近 儿 年 来 出 现 了 一 些 新 的 教育 形式 ， 如 网 上 授课 、 网 上 考试 等 。X xX 
X 高 校 是 一 所 师资 力量 非常 雄厚 的 学 校 ， 随 着 院 校 的 扩招 ， 学 生 数 量 不 断 增加 。 为 了 适应 新 形式 的 发 
展 ， 改 变 传统 的 教学 模式 ， 方 便 学 生 随 时 随地 地 对 自己 的 学 习 情 况 进 行 检 测 ， 减 轻 教师 的 工作 压力 ， 
现 委托 其 他 公司 开发 网 上 在 线 考试 系统 。 这 种 无 纸 的 网 络 考试 系统 ， 使 考 务 管 理 突破 时 空 限制 ， 提 高 
考试 工作 效率 和 标准 化 水 平 ， 使 学 校 管理 者 、 教 师 和 学 生 可 以 在 任何 时 候 、 任 何 地 点 通过 网 络 进行 考 
试 。 网 络 在 线 考试 系统 已 经 成 为 教育 技术 发 展 与 研究 的 方向 。 


4.2 系统 分 析 


4.2.1 需求 分 析 


随 着 计算 机 技术 的 发 展 和 网 络 技术 的 日 益 成 熟 ， 通 过 网 络 进行 信息 交流 已 成 为 一 种 快捷 的 交互 方 
式 。 在 这 种 网 络 环境 下 ， 学 校 或 考试 机 构 希 望 通过 建立 网 络 在 线 考试 网 站 来 扩大 知名 度 、 降 低 管理 成 
本 和 减少 人 力 、 物 力 的 投资 ， 从 而 为 考生 提供 更 全 面 、 更 灵活 的 服务 ， 并 全 面 、 准 确 地 对 考试 进行 跟 
踪 和 评价 。 与 此 同时 ， 考 生 希 望 根 据 自 己 的 学 习 情况 进行 测试 ， 并 能 够 得 到 客观 、 科 学 的 评价 ， 教务 
人 员 和 希望 能 够 有 效 地 改进 现 有 的 考试 模式 ， 提 高 考试 效率 。 

通过 实际 情况 的 调查 ， 要 求 网 络 在 线 考试 系统 具有 以 下 功能 : 
界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵 活 ， 树 立 企 业 形象 。 

要 求实 现在 线 考 试 功能 ， 自 动 核算 考试 成 绩 。 

要 求 提供 考试 时 间 倒 计时 功能 ， 使 考生 实时 了 解 考试 剩余 时 间 。 
要 求 系统 自动 阅卷 ， 保 证 考试 成 绩 真实 有 效 。 

要 求 考生 赁 准 考证 号 查询 考试 成 绩 ， 以 保证 信息 安全 。 

系统 运行 稳定 、 安 全 可 靠 。 

要 求 对 考生 及 考题 信息 进行 严格 管理 。 


4.2.2 ”可行 性 分 析 


加 


加 


办 办 办 多 轨 


可 行 性 分 析 的 目的 就 是 要 用 最 小 的 代价 在 尽 可 能 短 的 时 间 内 确定 问题 是 否 能 够 解决 。 通 过 分 析 解 
法 的 利弊 ， 来 判定 系统 目标 和 规模 是 否 现实 ， 系 统 完成 后 所 能 带 来 的 效益 是 否 达到 值得 去 投资 开发 这 
个 系统 的 程度 。 网 络 在 线 考试 系统 的 可 行 性 可 从 以 下 两 方面 考虑 。 


1. 经 济 可 行 性 


定期 地 组 织 考试 是 各 个 院 校 及 时 掌握 学 生 学 习 成 绩 的 有 效 方式 。 利 用 网 络 在 线 考试 系统 ， 一 方面 
可 以 节省 人 力 资源 ， 降 低 考 试 成 本 ; 另 一 方面 ， 在 线 考试 系统 能 够 快速 进行 考试 和 评分 ， 体 现 出 考试 
的 客观 与 公正 性 。 
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2. 技术 可 行 性 

开发 一 个 网 络 在 线 考试 系统 ， 涉 及 的 最 核心 的 技术 问题 就 是 如 何 实现 在 不 刷新 页 面 的 情况 下 实时 
显示 考试 时 间 及 剩余 时 间 ， 并 做 到 到 达 考 试 结束 时 间 时 自动 提交 试卷 的 功能 。 通 过 Ajax 技术 可 以 轻松 
实现 这 些 功 能 ， 这 为 网 络 在 线 考试 系统 的 开发 提供 了 技术 保障 。 


4.3 系统 设计 


4.3.1 系统 目标 


根据 前 面 所 作 的 需求 分 析 及 用 户 的 需求 可 知 ， 网 络 在 线 考试 系统 属于 中 小 型 软件 ， 在 系统 实施 后 
应 达到 以 下 目标 : 


回 ”采用 开放 、 动 态 的 系统 架构 ， 加 强 用 户 与 网 站 的 动态 交互 性 。 

回 具有 空间 性 。 被 授权 的 用 户 可 以 在 异地 登录 网 络 在 线 考试 系统 ， 无 须 到 指定 地 点 进行 考试 。 

回 ”操作 简单 方便 、 界 面 简洁 美观 。 

回 ”系统 提供 考试 时 间 倒 计时 功能 ， 使 考生 实时 了 解 考试 剩余 时 间 。 

回 ”随机 抽取 试题 。 

回 ”实现 自动 提交 试卷 的 功能 。 当 考试 时 间 到 达 规 定时 间 时 ， 如 果 考 生还 未 提交 试卷 ， 系 统 将 自 
动 交 卷 ， 以 保证 考试 严肃 、 公 正 地 进行 。 

回 系统 自动 阅卷 ， 保 证 成 绩 真实 准确 。 

回 ”考生 可 以 查询 考试 成 绩 。 

回 ”对 考生 注册 信息 进行 管理 。 

回 “系统 运行 稳定 、 安 全 可 靠 。 


4.3.2 ”系统 功能 结构 


网 络 在 线 考试 系统 的 前 台 功能 结构 如 图 4.1 所 示 。 
网 络 在 线 考试 系统 的 后 台 功 能 结构 如 图 4.2 所 示 。 


网 络 在 线 考试 系统 的 前 台 


网 络 在 线 考试 系统 的 前 台 


考 
生 
信 
奶 
管 
理 


滑 吐 尝 闪 区 毕 
EEETEE 
内 哄 证 坟 陆 必 
性 汪 


图 4.1 网 络 在 线 考试 系统 的 前 台 功 能 结构 图 图 4.2 ”网络 在 线 考试 系统 的 后 台 功能 结构 图 
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4.3.3 ”系统 流程 图 


为 了 读者 能 够 更 好 地 学 习 ， 下 面 给 出 网 络 在 线 考试 网 的 网 站 流程 概述 及 系统 流程 图 。 

考生 在 网 络 在 线 考试 系统 中 通过 注册 为 网 站 用 户 ， 登 录 网 站 进行 相关 操作 。 考 生 登 录 后 ， 可 以 进 
行 在 线 考试 、 查 询 成 绩 和 修改 个 人 密码 的 操作 。 在 考试 前 ， 考 生 需 要 阅读 考试 规则 、 选 择 考试 套 题 后 
开始 考试 ; 考试 时 间 结 束 时 ， 考 生 提交 试卷 ;提交 试卷 的 同时 ， 系 统 将 自动 返回 本 次 考试 的 考试 结果 。 
网 站 的 管理 员 通 过 登录 模块 可 以 登录 到 网 站 的 后 台 系统 ， 对 考生 信息 、 考 试 信息 、 管 理 员 信息 进行 管 
理 。 网 络 在 线 考试 系统 的 系统 流程 如 图 4.3 所 示 。 


图 4.3 系统 流程 图 


4.3.4 系统 预览 


网 络 在 线 考试 系统 由 多 个 页 面 组 成 ， 下 面 仅 列 出 儿 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 

考生 登录 页 面 如 图 4.4 所 示 ， 该 页 面 主要 用 于 考生 登录 , 实现 在 线 考试 及 考试 成 绩 查询 等 功能 。 在 
线 考试 页 面 如 图 4.5 所 示 , 该 页 面 用 于 实现 在 线 答题 功能 , 同时 提供 了 显示 考试 时 间 和 剩余 时 间 及 自动 
提交 试卷 功能 。 


(lic san xm ea seen ass las 


图 4.4 考生 登录 (光盘 \TM\04\online ks\denglu.php 图 4.5 在 线 考试 (光盘 \TM\04\online_ks\ksks.php) 


第 4 章 网 络 在 线 考试 系统 (Apache+PHP+Ajax+SQL Server 2000 实现 ) ”毛毛 所 


考题 类 别管 理 页 面 如 图 4.6 所 示 , 该 页 面 主 要 用 于 实现 显示 考题 类 别 的 基本 信息 、 添加 考题 类 别 和 
删除 考题 类 别 等 功能 。 考 题 信息 管理 页 面 如 图 4.7 所 示 ， 该 页 面 主要 用 于 管理 考试 题目 信息 。 


图 4.6 考题 类 别管 理 (光盘 \TM\04\…\admin\ktlb_gl.php) 图 4.7 考题 信息 管理 (光盘 \TM\04\…\admin\ktxx_gl.php) 
到 
说 明 由 于 路 径 太 长 ， 因 此 省 略 了 部 分 路 径 ， 省 略 的 路 径 是 “online_ksV”。 


4.3.5 ”开发 环境 


在 开发 网 络 在 线 考试 系统 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows Server 2003/Linux (推荐 )。 
服务 器 : Apache 2.2.8。 

PHP 软件 : PHP 5.2.6。 

数据 库 : SQL Server 2000。 

开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辩 率 : 最 佳 效果 为 1024x768 像素 。 


客户 端 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辩 率 : 最 佳 效果 为 1024x768 像素 。 
4.3.6 文件 夹 组 织 结构 


在 编写 代码 之 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创建 出 来 〈 例 如 ， 创 建 一 个 名 为 images 的 文 
件 夹 ， 用 于 保存 网 站 中 所 使 用 的 图 片 )， 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 也 可 以 规范 网 站 的 整体 架 
构 。 笔 者 在 网 络 在 线 考试 系统 中 设计 了 如 图 4.8 所 示 的 文件 夹 组 织 结构 图 。 在 开发 时 ， 只 需要 将 所 创建 


的 文件 保存 在 相应 的 文件 夹 中 即 可 。 


固 加 六 图 图 回回 图 图 加 
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一 用 下 结 网站 后 E 文 人 


一 一 一 一 一 一 一 一 局 的 


图 4.8 网 络 在 线 考试 系统 的 文件 夹 组 织 结构 
4.4 数据 库 设计 


4.4.1 数据库 分 析 


由 于 网 络 在 线 考试 系统 对 于 数据 的 安全 性 及 完整 性 要 求 比 较 高 ， 并 且 为 了 增加 程序 的 适用 范围 ， 
还 要 保证 系统 可 以 拥有 存储 足够 多 数据 的 能 力 。SQL Server 2000 是 一 种 高 性 能 的 关系 型 数据 库 管理 系 
统 ， 不 仅 安全 可 靠 ， 而 且 易 于 操作 ， 己 成 为 在 线 事务 进程 和 数据 仓库 等 最 好 的 数据 库 平台 之 一 。 综 上 
所 述 ， 本 系统 采用 SQL Server 2000 数据 库 。 


4.4.2 数据 库 概念 设计 

根据 以 上 各 节 对 系统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 使 用 的 数据 库 实体 分 别 为 考生 信 
息 实 体 、 管 理 员 实体 、 考 题 类 别 实 体 和 考题 信息 实体 。 下 面 分 别 介绍 这 几 个 实体 的 E-R 图 。 

1. 考生 信息 实体 

考生 信息 实体 包括 编号 、 考 生 姓 名 、 联 系 方式 、 准 考证 号 、 考 试 成 绩 、 考 题 类 别 、 考 试 时 间 、 考 
试 状态 和 考生 密码 等 属性 。 考 生 信息 实体 的 E-R 图 如 图 4.9 所 示 。 

2. 管理 员 实 体 


管理 员 实 体 包括 编号 、 管 理 员 名 称 和 密码 等 属性 。 管 理 员 实体 的 E-R 图 如 图 4.10 所 示 。 


图 4.9 考生 信息 实体 E-R 图 图 4.10 管理 员 实体 E-R 图 
3. 考题 类 别 实 体 
考题 类 别 实体 包括 编号 和 类 别名 称 等 属性 。 考 题 类 别 实 体 的 E-R 图 如 图 4.11 所 示 。 
4. 考题 信息 实体 
考题 信息 实体 包括 编号 、 考 题 类 别 、 考 试 成 绩 、 考 题 内 容 、 考 题 答案 、 正 确 答 案 及 所 属 套 题 等 属 
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性 。 考 题 信息 实体 的 E-R 图 如 图 4.12 所 示 。 


Cas 考试 成 线 
考题 类 别 所 局 套 大 | 考题 信息 | 各 题 内 容 》 
> < < 
图 4.11 考题 类 别 实体 ER 图 图 4.12 考题 信息 实体 E-R 图 
4.4.3 数据库 物理 结构 
根据 4.4.2 节 的 数据 库 概念 设计 , 可 以 创建 与 实体 对 
应 的 数据 表 。 为 了 使 读者 对 本 系统 数据 库 的 结构 有 一 个 | 一 掉 tb_wer( 壮 生 信 息 才 ) 
更 清晰 的 认识 ， 下 面 给 出 数据 库 中 所 包含 数据 表 的 结构 二 人 ee ( 伟 理 页 信 息 表 ) 


| 一 日 ”tb_kth (考题 类 别 信息 表 ) 
一 一 日 ”tb_kt (考题 信息 表 ) 


图 4.13 db_online 数据 库 所 包含 数据 表 的 结构 图 


图 ， 如 图 4.13 所 示 。 
本 系统 共 包含 4 张 数据 表 ， 下 面 进行 详细 介绍 。 


1. tb_user (考生 信息 表 ) 

考生 信息 表 用 来 保存 考生 信息 。 该 数据 表 结 构 如 图 4.14 所 示 。 

2. tb_admin (管理 员 信 息 表 ) 

管理 员 信 息 表 用 来 保存 管理 员 的 用 户 名 和 密码 。 该 数据 表 结 构 如 图 4.15 所 示 。 


CE 
四 vl 时 
印 兰 主 姓名 
印 等 生 联系 方式 
| din 
m vv Er 列 名 | 数据 天 型 | 长 度 [下 许 空 | 标识 | 标识 种 子 ] “手术 <“ 
外 v ee mlid int 4 Pag Es 
= varchs 20 V 
全 3 a mV 管理 员 密码 
"4 
图 4.14 考生 信息 表 结 构 图 4.15 管理 员 信 息 表 结 构 


3. tb_ktlb (考题 类 别 信息 表 ) 

考题 类 别 信息 表 用 来 保存 考题 类 别 。 该 数据 表 结 构 如 图 4.16 所 示 。 

4. tb_kt (考题 信息 表 ) 

考题 信息 表 用 来 保存 考试 题目 信息 和 考题 答案 等 相关 信息 。 该 数据 表 结 构 如 图 4.17 所 示 。 


tb_ktlb 
列 名 数据 类 型 | 长 度 | 区 许 宝 | 标识 [标识 种 子 | 撕 述 | 
lth id int 4 vl 自动 篇 号 
online ktlb varchar 50 考 本 类 别 
到 


图 4.16 考题 类 别 信息 表 结 构 
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4.5.1 前 台 首 页 概述 


考生 通过 考生 登录 模块 的 验证 后 ， 可 以 登录 到 网 络 在 线 考试 系统 的 前 台 首 页 。 前 台 首 页 主要 用 于 
实现 前 台 功能 导航 ， 该 页 面 主要 包括 “考生 注册 ”“ 考 生 登录 “修改 密码 “成 绩 查 询 ^“ 进 入 考 
场 ” 和 “退出 信息 ”6 个 导航 链接 。 

本 案例 中 提供 的 前 台 首 页 如 图 4.18 所 示 。 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\04\online_ks\index.php。 


[ioNE coM ?ssrum | sn | Naeem | mawm | ia 看 | Pd 


0 Bee 


图 4.18 网络 在 线 考试 系统 前 台 首 页 
4.5.2 前台 首页 技术 分 析 
PHP 连接 SQL Server 数据 库 的 方法 有 两 种 : 一 是 通过 ADO 方式 连接 ; 二 是 通过 mssql_connect() 
函数 连接 。 下 面 分 别 介绍 这 两 种 连接 方式 。 
1. 通过 ADO 方式 连接 SQL Server 数据 库 


利用 ADO 方式 建立 与 SQL Server 数据 库 的 连接 ， 与 使 用 ADO 方式 建立 与 Access 数据 库 连 接 的 
方法 类 似 ， 建 立 连接 是 在 本 实例 的 conn.php 文件 中 实现 的 。 代 码 如 下 : 

<?php 

$conn=new com("adodb.connection"); 

$connstr="provider=sqloledb;data source=localhost;uid=sa;pwd=;database=db_online"; 

$conn->open($connstr); 

?> 


2. 通过 mssql_connect() 函 数 连 接 SQL Server 数据 库 


通过 mssql_connectO 函 数 建立 与 SQL Server 数据 库 的 连接 首先 应 该 配置 php.ini 文件 , 删除 “extension= 
php_mssql.dl1” 前 面 的 注释 ， 然 后 重新 启动 Apache 服务 器 。 通 过 mssql_connect() 函 数 建立 与 数据 库 连 
接 的 代码 如 下 : 


G@ 
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<?php 
©@ $conn=mssql_connect("localhost',"sa",""); /建立 与 SQL Server 数据 库 的 连接 
@ mssql_select_db("db_online ",$conn); /选择 数据 库 
?> 
< 代 码 贴 十 


@ mssql connect(0) 函 数 : 打开 一 个 到 SQL Server 服务 器 的 连接 。 如 果 成 功 ， 则 返回 一 个 SQL Server 连接 标识 ; 失 
败 则 返回 false。 语 法 如 下 : 


resource mssql_connect ( [string servemame [, string username [, string password [, bool new_link]]]] ) 

参数 servemame 指 SQL Server 服务 器 ; 参数 usernamer 指 用 户 名 ; 参数 password 指 用 户 密 码 ; 参数 new_link 指 连 
接 标识 。 

@ mssql select_db() 函 数 : 选择 SQL Server 数据 库 。 如 果 成 功 ， 则 返回 true; 失败 则 返回 false。 语 法 如 下 : 


bool mssql_select db( string database_name [,resource link_identifier]) 


参数 database_name 指数 据 库 名 称 ; 参数 link_identifier 指 连接 标识 。 
4.5.3 前台 首页 的 实现 过 程 


网 络 在 线 考试 系统 前 台 首 页 主要 实现 了 考生 登录 功能 ， 考 生 通 过 准 考证 号 和 密码 进行 登录 。 
考生 登录 页 面 所 涉及 的 重要 表单 元 素 如 表 4.1 所 示 。 


表 4.1 考生 登录 页 面 所 涉及 的 重要 表单 元 素 


| 元 素 类 型 | 重要 属性 | 
method="post" action="denglu ok.php" 


online number id="online number" size="20" 


online pass | password | id="online pass" size="20" 
Submit | suomit | value=" 提 交 " 


考生 在 考生 登录 页 面 输 入 准确 的 准 考证 号 和 密码 后 ， 单 击 “ 提 交 ” 按 钮 ， 提 交 表单 信息 到 数据 处 
理 页 , 应 用 mssql_query() 函 数 检索 考生 是 否 成 功 登录 。 如 果 查 询 结 果 为 真 , 则 将 准 考证 号 存储 在 session 
变量 中 ， 耕 则 弹出 提示 信息 。 代 码 如 下 

倒 程 01 ”代码 位 置 ， 光盘 \TM\04\online_ks\denglu_ok.php 


<?php 

include("conn/conn.php"); /| 连接 数据 库 文件 
S$online_number=$_POST[online_number]; // 获 取 准 考证 号 
S$online_pass=$_POST[online_pass]; // 获 取 考 生 密 码 


1 eveexxxxexxxexexexxwtwwstwws 检 索 考 生 的 准 考证 号 和 密码 是 否 正确 seeeeeeseeeexxesexeeeexeeweat 


@ Squery=mssql_query("select * from tb_user where online_number='$online number and 
online_pass='$online_pass"); 
@ if(mssql_num_rows($query)>0){ // 如 果 查 询 结 果 为 真 
© session_register(online_number); // 将 准 考证 号 存储 在 session 变量 中 
echo "<script> alert(' 登 录 成 功 !"); window.location.href='index.php?online= 考 试 规则 ';</script>"; 
}else{ // 如 果 没有 检索 到 数据 


Re) 
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echo "登录 失败 !"; // 弹 出 提示 信息 
3 


Ty 
< 代码 贴 二 
@ mssql query: 向 与 指定 的 SQL Server 服务 器 中 的 当前 活动 数据 库 发 送 一 条 SQL 查询 。 语 法 如 下 : 


resource mssql_query ( string query [, resource link_identifier] ) 


参数 query 传 入 的 是 SQL 的 指令 ; 参数 link identifier 传 入 的 是 由 mssql_connect() 函 数 或 mssql_pconnect() 函 数 返回 
的 连接 号 ， 如 果 省 略 该 参数 ， 则 会 使 用 最 后 一 个 打开 的 MySQL 数据 库 连 接 。 

@ mssql_ num_ rows: 获取 结果 集中 行 的 数目 。 值 得 注意 的 是 ， 该 函数 仅 对 SELECT 语句 有 效 。 

@ session_register: 在 整个 域 范围 内 增加 一 个 session 变量 。 语 法 如 下 : 


boolean session_register(string name) 


参数 name 用 于 指定 新 session 变量 的 名 称 。 
4.6 考生 信息 模块 设计 
4.6.1 考生 信息 模块 概述 


考生 信息 模块 主要 包括 考生 注册 、 考 生 登录 、 修 改 密码 3 个 功 
能 。 考 生 首先 要 注册 为 网 站 用 户 ， 然 后 才 被 授权 登录 网 站 进行 一 系 
列 操作 的 权限 ; 登录 后 考生 还 可 以 修改 个 人 的 密码 。 考 生 信息 模块 
的 系统 流程 如 图 4.19 所 示 。 


4.6.2 考生 信息 模块 的 技术 分 析 


在 考生 注册 信息 模块 中 应 用 到 Ajax 无 刷新 技术 获取 考生 的 准 
考证 号 码 和 密码 。 在 介绍 该 功能 的 实现 过 程 之 前 ， 先 对 Ajax 技术 
中 XmlHttpRequest 对 象 的 属性 和 方法 进行 详细 讲解 。 

XmlHttpRequest 对 象 是 Ajax 技术 的 核心 ， 有 关 该 对 象 的 属性 
和 方法 的 详细 介绍 如 表 4.2 和 表 4.3 所 示 。 


表 4.2 XmlHttpRequest 对 象 的 属性 


图 4.19 考生 信息 模块 的 系统 流程 图 


属 性 描述 
readystate 返回 当前 的 请 求 状态 
onreadystatechange 当 readyState 属性 改变 时 就 可 以 读 取 此 属性 值 
status 返回 Http 状态 码 
responseText 将 返回 的 响应 信息 用 字符 串 表示 
ResponseBod. 返回 响应 信息 正文 ， 格 式 为 字 节 数组 
ResponseXML 将 响应 的 domcoment 对 象 解析 成 XML 文档 并 返 | 
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表 4.3 XmlHttpRequest 对 象 的 方法 


方 ” 法 描述 
open() 初始 化 一 个 新 请 求 
send0 | 发 送 请 求 
GetAllReponseHeaders() | 返回 所 有 Http 头 信息 
GetResponseHearder 返回 指定 的 Http 头 信息 
SetRequestHeader() 添加 指定 的 Http 头 信 息 
Abort 停止 当前 的 Http 请 求 


1. readystate 属性 
readystate 属性 用 于 返回 当前 的 请 求 状 态 ， 请 求 状 态 共 有 5 种 ， 如 表 4.4 所 示 。 
表 4.4 readystate 属性 值 


属 性 值 描述 
0 表示 尚未 初始 化 ， 即 未 调用 open 方法 
1 建立 请 求 ， 但 还 未 调用 send 方法 发 送 请 求 
2 发 送 请求 
3 处 理 请 求 
4 完成 响应 ， 返 回 数据 
2. status 属性 
status 属性 用 于 返回 Http 状态 码 ， 常 用 Http 状态 码 如 表 4.5 所 示 。 
表 4.5 ”Http 状态 码 
属 性 名 描述 
200 操作 成 功 
404 没有 发 现 文件 
500 | 服务 器 内 部 错误 
505 服务 器 不 支持 或 拒绝 请 求 中 指定 的 Http 版 本 


3. responseText 属性 


responseText 属性 将 返回 的 响应 信息 用 字符 串 来 表示 。 在 默认 情况 下 ,返回 的 响应 信息 的 编码 格式 
为 utf-8。 


4. responseXML 属性 

responseXML 属性 用 于 将 响应 的 domcoment 对 象 解析 成 XML 文档 并 返回 。 
5. open 方 法 

open 方法 用 于 初始 化 一 个 新 的 请 求 。 
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语法 : 
open(String method, String url, Boolean asyn, String user String password) 


其 中 ，method 和 url 是 必 选 参数 ，asyn、user 和 password 是 可 选 参数 。open() 方 法 各 参数 说 明 如 表 4.6 
所 示 。 
表 4.6 open() 方 法 参数 


参数 名 称 描述 


method | 指明 了 新 请 求 的 调用 方法 ， 其 取 值 有 get 和 post 

ul 表示 要 请 求 页 面 的 URL 地 址 。 格 式 可 以 是 相对 路 径 、 绝 对 路 径 或 网 络 路 径 
asyn 说 明 该 请 求 是 异步 传输 还 是 同步 传输 ， 默 认 值 为 true (允许 异步 传输 ) 
user | 服务 器 验证 时 的 用 户 名 


Password 服务 器 验证 时 的 密码 


6. send() 方 法 


send() 方 法 用 于 发 送 请 求 到 服务 器 。 


send(body) 


如 果 没 有 要 发 送 的 内 容 ， 则 body 可 以 省 略 或 为 Null。 
7. GetAllReponseHeaders() 方 法 


GetAllReponseHeaders() 方 法 用 于 获取 响应 的 所 有 Http 头 信 息 。 获 取 到 的 信息 是 按照 “名 称 / 键 值 ” 
排列 的 ， 信 息 之 间 是 用 “: ”冒号 进行 分 隔 的 。 
语法 : 


GetAllReponseHeaders() 


8. GetResponseHeader() 方 法 

GetResponseHeader() 方 法 是 获取 响应 中 指定 的 Http 头 信息 。 
语法 : 

GetResponseHeader(String head) 

9. SetRequestHeader() 方 法 

SetRequestHeader() 方 法 用 于 添加 一 个 Http 头 信息 。 

语法 : 

SetRequestHeader(name,value) 


其 中 ，name 表示 Http 头 名 称 ，value 表示 信息 内 容 。 
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10. Abort() 方 法 
Abort(0 方 法 用 于 取消 一 个 请 求 。 
语法 : 


Abort() 


4.6.3 考生 注册 的 实现 过 程 


国 考生 注册 使 用 的 数据 表 : tb_user 
在 网 络 在 线 考试 系统 首页 ， 单 击 “ 考 生 注 册 ” 超 链接 ， 即 可 进入 考生 注册 页 面 ， 输 入 个 人 的 真实 
资料 后 ， 单 击 “ 注 册 ” 按 钮 ， 系 统 将 根据 输入 的 个 人 资料 自动 生成 准 考证 号 和 考生 密码 ， 如 图 4.20 所 示 。 


考生 证 里 | 必 生 型 录 | 人 本 码 | 所 和 济 | 退 入 考生 “| 严 由 各 


在 线 考试 争 统 


aaa 

LE Em 

ETT EE 
| 


Od 


图 4.20 考生 注册 页 面 的 运行 结果 
oY 
说 明 考生 需要 通过 准 考证 号 和 密码 登录 网 络 在 线 考试 的 核心 模块 ， 即 在 线 考试 及 成 绩 查 询 等 。 


在 考生 注册 页 面 zhuce.php 页 中 实现 考生 注册 信息 的 提交 ， 并 且 通 过 Ajax 的 页 面 无 刷新 技术 获取 
考生 的 准 考 证 号 和 密码 。zhuce.php 页 中 的 关键 代码 如 下 : 
倒 程 02 ”代码 位 置 ， 光盘 \TM\04\online_ks\zhuce.php 
<!-- 调 用 一 个 JavaScript 脚本 文件 --> 
<script type="text/javascript" src="js/zhuce_js.js"></script> 
<!-- 通 过 onClick 事件 调用 一 个 自 定义 函数 --> 
<input type="button" name="Submit" value=" 注 册 " onClick="process()"> 
<!-- 输 出 一 个 div 标签 中 的 值 --> 
<div id="divMessage" /> 


在 zhuce_js.js 文件 中 首先 定义 一 个 createXmlHttpRequestObject() 对 象 ， 并 获取 XMLHttpRrequest 
对 象 ， 然 后 定义 xmlHttp 用 来 存储 将 要 使 用 的 XMLHttpRrequest 对 象 。 关 键 代码 如 下 : 
倒 程 03 ”代码 位 置 ， 光盘 \TM\04\online_ks\js\zhuce _jsjs 


var xmlHttp=createXmlHttpRequestObject(); /定义 XMLHttpRrequest 对 象 
function createXmlHttpRequestObject(){ /获取 XMLHttpRrequest 对 象 


_- 国 


PHP 项 目 开发 全 程 实录 (第 3 版 ) 


var xmlHttp; 
if(window.ActiveXObjectX{ 
try{ 


// 用 来 存储 将 要 使 用 的 XMLHttpRrequest 对 象 
/如果 在 internet Explorer 下 运行 


xmlHttp=new ActiveXObject("Microsoft.XMLHTTP"); 


}catch(eX{ 
xmlHttp=false; 


} 
jelse{ 
try{ 
xmlHttp=new XMLHttpRequest(); 
j}catch(e)f 
xmlHttp=false; 
2 


和 

if(IxmlHttp) 
alert(" 返 回 创建 的 对 象 或 显示 错误 信息 "); 
else 
return xmlHttp; 


} 


1/ 如果 在 Mozilla 或 其 他 的 浏览 器 下 运行 


// 返 回 创建 的 对 象 或 显示 错误 信息 


然后 使 用 XMLHttpRequest 对 象 创建 异步 Http 请 求 ， 定 义 函数 process()， 对 表单 中 提交 的 数据 进 
行 判 断 ， 并 且 获 取 表单 中 输入 的 信息 ， 在 服务 器 端 执行 zhuce_ok.php 文件 ， 向 服务 器 发 送 请 求 。 关 键 


代码 如 下 : 
倒 程 04 代码 位 置 ， 光盘 \TM\04\online_ks\js\zhuce_jsjs 


function process(X{ 
if(form1.usermame.value==""){ 
alert(" 请 输入 姓名 ! "); 
form1.username.select(); 
retum(false); 


WN ‘tel.value=="™"){ 
alert(" 请 输入 电话 号 码 !"); 
form1 .tel.select(); 
return(false); 


PR .tel.value)!=trueX{ 
alert(" 您 输入 的 电话 号 码 的 格式 不 正确 !"); 
form1 .tel.select(); 
return(false); 


if(form1.address.value=="" 
alert(" 请 输入 联系 地 址 !"); 
form1.address.select(); 
return(false); 


} 
if(xmlHttp.readyState==4 || xmlHttp.readyState==0){ 
names = document.getElementByld("usemame").value; 
tels = document.getElementByld("tel").value; 
addresss =document.getElementByld("address").value; 
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// 使 用 XMLHttpRequest 对 象 创建 异步 Http 请 求 
// 如 果 考 生 姓 名 为 空 

// 则 弹出 提示 

1/ 返回 焦点 到 考生 姓名 文本 框 


// 如 果 考 生 联系 电话 为 空 
// 则 弹出 提示 
// 返 回 焦 点 到 考生 联系 方式 文本 框 


// 如 果 考生 联系 方式 格式 不 正确 
// 则 弹出 提示 
// 返 回 焦点 到 考生 联系 方式 文本 框 


// 如 果 考 生 联 系 地址 为 空 
// 则 弹出 提示 
// 返 回 焦 点 到 考生 联系 地 址 文本 框 


// 在 xmlHttp 对 象 不 忙 时 进行 处 理 
// 获 取 用 户 在 线 表单 中 输入 的 姓名 
// 获 取 用 户 在 线 表单 中 输入 的 电话 
// 获 取 用 户 在 线 表单 中 输入 的 地 址 


第 4 章 网络 在 线 考试 系统 (Apache+PHP+Ajax+SQL Server 2000 实现 ) 


xmlHttp.open("GET","zhuce_ok.php?online_user="+names+"& online_tel="+tels+"& 


online_address="+addresss,true); // 在 服务 器 端 执行 zhuce_ok.php 
xmlHttp.onreadystatechange=handleServerResponse; /定义 获取 服务 器 端 响应 的 方法 
xmlHttp.send(null); // 向 服务 器 发 送 请 求 
jelse 
setTimeout('process(),1000); // 如 果 服 务 器 忙 ，1 秒 后 重 试 


最 后 获取 从 服务 器 端 返回 的 消息 。 关 键 代码 如 下 : 
倒 释 05 ”代码 位 置 ， 光盘 \TMVO4\online_ ksVsvzhuce_jsjs 


// 当 收 到 服务 器 端的 消息 时 自动 执行 
function handleServerResponse(){ 
if(xmlHttp.readystate==4X{ // 在 处 理 结束 时 进入 下 一 步 
if(xmlHttp.status==200){ // 状 态 为 200 表示 处 理 成 功 结束 
xmlResponse=xmlHttp.responseXML; // 获 取 服 务 器 端 发 来 的 XML 信息 


xmlDocumentElement=xmlResponse.documentElement; // 获 取 XML 中 的 文档 对 象 〈 根 对 象 ) 
helloMessage=xmlDocumentElement.firstChild.data; // 获 取 第 一 个 文档 子 元 素 的 文本 信息 
// 使 用 从 服务 器 端 发 来 的 消息 更 新 客户 端 显示 的 内 容 
document.getElementByld("divMessage").innerHTML="<i>'+helloMessage+'</i>"; 
setTimeout('process()',1000); /重新 开始 


}else{ 
// 如 果 Http 的 状态 不 是 200 表示 发 生 错 误 
alert("There was a problem accessing the server:"+xmlHttp.statusText); 


} 


} 

zhuce_ok.php 文件 在 服务 器 端 被 执行 ， 将 获取 到 的 用 户 信息 和 随机 生成 的 准 考证 号 、 密 码 添 加 到 
数据 库 中 ， 并 且 将 准 考证 号 和 密码 返回 到 考生 注册 页 面 中 。 关 键 代 码 如 下 : 

倒 程 06 ”代码 位 置 ， 光盘 \TM\04\online_ks\zhuce_ok.php 


<?php 

header('Content-Type: text/xml'); 1/ 创建 一 个 XML 格式 输出 
echo '<?xml version="1.0" encoding="gb2312" standalone="yes" ?>"; /创建 XML 头 

echo '<response>'; /创建 <response> 元 素 

上 echo $online_user=$_GET[online_usen]; // 获 取 考生 姓名 
$online_tel=$_GET[online_tel]; // 获 取 考 生 联系 电话 
S$online_address=$_GET[online_address]; // 获 取 考 生 联 系 地 址 

@ $online_number=substr(mt_rand(100000,999999),0,6); // 生 成 6 位 随机 的 准 考证 号 码 
$online_pass=substr(mt_rand(100000,999999),0,6); // 生 成 6 位 随机 的 考生 密码 


/cxxxsesxtkxstkyxesttxssstx 根 据 从 客户 端 获取 的 用 户 创建 输出 **ssserseseeessessestssesxses il 
include("conn/conn.php"); 
$query=mssql_query("insert into tb_user(online_user,online_tel,online_address,online_number,online_pass) 
values('$online_user','$online_tel','$online_address','$online_number','$online_pass')"); 
if($query==trueX{ 

echo $online_user; 


echo "用 户 注册 成 功 ， 这 是 您 的 准 考 证 号 码 $online_number. 和 密码 $online_pass."; 


_ 国 ) 
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} 
echo '</response>'; /| 关闭 <response> 元 素 
ts 
< 代 码 贴 十 
@ $_GET 预 定义 变量 : PHP 使 用 $_GET 预定 义 变量 自动 保存 通过 get 方法 传 过 来 的 值 ， 使 用 格式 为 : 


$_GET[namel] 
@ substr() 函 数 : 从 指定 的 字符 串 中 按照 指定 的 位 置 截 取 一 定 长 度 的 字符 。 语 法 如 下 : 
string substr(string str,int start,int length) 


参数 str 指定 字符 囊 对 象 。 参数 start 指定 开始 截取 字符 串 的 位 置 ， 如 果 为 负数 ， 则 从 字符 囊 的 末尾 开始 截取 。 参数 
length 是 可 选 参数 ， 指 定 截取 字符 的 个 数 ， 如 果 为 负数 ， 则 表示 取 到 倒数 第 length 个 字符 。 


4.6.4 单元 测试 


在 应 用 Ajax 技术 实现 在 线 考试 的 用 户 注册 模块 后 , 为 保证 程序 能 够 正常 运行 , 必须 对 其 进行 测试 。 
考生 注册 模块 的 运行 结果 如 图 4.21 所 示 。 
[ac 
八 I 以 后 ,双击 显示 在 状态 加 
网 页 包 合 司 误 时 她 终 显 示 访 信息 人) 


上“ 和 证 | eane wo) < 


行 ， 84 

字符 :4 

错误 :缺少 对 象 
代码 :0 

VRL; http-//127.0.0 1/Tsoft/online_ks/index. php?online= 用 户 注册 


上 -个 四 TD 


图 4.21 考生 注册 模块 运行 出 现 的 错误 


从 弹出 的 错误 信息 来 看 , 是 程序 中 与 JavaScript 脚本 有 关 的 内 容 出 现 了 问题 。 经 过 反复 测试 和 查找 
发 现 ， 问 题 不 是 出 现在 zhuce_ js.js 文件 中 ， 因 为 在 该 文件 中 定义 的 自 定 义 函 数 process() 是 可 以 运行 的 ， 
所 以 问题 应 该 是 出 现在 对 服务 器 返回 的 信息 进行 输出 中 ， 即 zhuce_ok.php 文件 中 。 对 该 文件 进行 分 析 
发 现 ， 原 来 在 应 用 Ajax 技术 时 ， 服 务 器 端 返回 的 消息 采用 的 是 XML 格式 ， 可 是 在 客户 端 接收 XML 
文档 时 却 忽略 了 “header(Content-Type: text/xml);” 头 部 信息 的 使 用 ， 从 而 导致 客户 端 用 于 解析 XML 
的 API 抛 出 一 个 错误 信息 。 只 有 在 设置 头 部 信息 (header('Content-Type: texyxml);) 后 ， 客 户 端 才 可 以 
正确 地 接收 到 服务 器 端 传递 的 XML 文档 信息 。 正 确 的 程序 代码 如 下 : 

<?php 

1/ 创建 一 个 XML 格式 输出 

header('Content-Type: text/xml"); /标记 输出 的 是 XML 文档 

// 创 建 XML 头 


echo '<?xml version="1.0" encoding="gb2312" standalone="yes" ?>"; 
?> 
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4.7 在 线 考试 模块 设计 


4.7.1 在 线 考试 模块 的 概述 


在 线 考 试 模块 的 主要 功能 是 允许 考生 在 网 站 上 针对 指定 的 课程 进行 考试 。 在 该 模块 中 ， 考 生 首 先 


需要 登录 到 本 系统 中 ， 阅 读 考 试 规则 ， 在 同意 所 列 出 的 考试 规则 
后 ， 才 能 选择 考试 类 别 和 套 题 ， 然 后 进入 考试 页 面 进行 答题 ， 当 
考生 提交 试卷 或 者 到 达 考试 结束 时 间 时 ， 系 统 将 自动 对 考生 提交 
的 试卷 进行 评分 ， 并 给 出 最 终 考试 成 绩 。 在 线 考 试 模块 的 系统 流 
程 如 图 4.22 所 示 。 


4.7.2 在线 考试 模块 的 技术 分 析 


在 线 考试 模块 中 最 核心 的 功能 是 如 何 输出 考题 、 如 何 对 提交 
的 答案 进行 判断 ， 并 且 将 答案 输出 到 当前 页 中 。 由 于 上 述 3 个 方面 
的 内 容 都 是 在 同一 页 面 中 显示 , 所 以 在 技术 实现 过 程 中 这 3 个 方面 
的 内 容 是 以 一 个 整体 形式 出 现 的 。 这 里 以 单 选 题 为 例 进 行 讲解 。 

首先 以 上 一 页 表单 中 提交 的 值 为 条 件 ， 执 行 查询 语句 ， 通 过 
while 循环 语句 输出 查询 结果 ; 然后 根据 输出 的 查询 结果 创建 单 选 
按钮 组 ， 生 成 考题 ， 最 后 ， 将 考题 答案 提交 到 本 页 ， 对 提交 的 答 
案 与 数据 库 中 存储 的 正确 答案 进行 比较 ， 并 且 将 答案 输出 到 本 页 
中 。 单 选 题 输出 、 答 案 判 断 和 考题 答案 输出 的 关键 代码 如 下 : 

倒 程 07 ”代码 位 置 ， 光盘 \TM\04\online_ks\ksks.php 


是 
选择 考题 类 别 和 套 题 


提交 试卷 /到 达 考试 结束 时 间 
显示 本 次 考试 成 绩 


4.22 在线 考试 模块 的 系统 流程 图 


<?php 


/* 根 据 提交 的 变量 执行 查询 语句 ， 从 数据 库 中 读 取 数据 ， 指 定 考题 的 类 型 为 0， 表 示 为 单 选 题 */ 
$query0=mssql_query("select * from tb_kt where kt_Ib="$kt_lbes' and kt_Ix="0' and kt_small_lb='$kt_small_Ib™); 


$x=1; // 定 义 变量 $x=1， 用 于 输出 题 号 
S$fen0=0; // 定 义 变量 $fen0=0， 用 于 考试 分 数 的 计算 
while($myrow0=mssql_fetch_array($query0)X /| 执行 while 循环 语句 

是 

<tr> 


<!-- 输 出 单 选 题 的 题 号 、 考 题 内 容 以 及 该 题 的 分 数 --> 


<td width="443" height="20" bgcolor="#FFFFFF">&nbsp;&nbsp;<?php echo $x.".".$myrowO["kt_nr"]?> </td> 


<td bgcolor="#FFFFFF"><?php echo $myrowO[kt_fs];?></td> 
< 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 > 

</tr> 

<?php 


~~: 获取 考题 提交 的 答案 ， 将 提交 的 考题 答案 进行 分 着， 并 赋 给 变量 $str0*…***/ 


$array0=explode(™™, $myrowO["kt_daan"]); 
if($_POST[Submit]!="" 
for($a=0;$a<count($array0);$a++){ 


// 将 数据 库 中 的 考题 答案 以 * 进 行 分 割 , 并 写 入 到 数组 中 
// 判 断 表 单 提交 是 否 为 空 
// 循 环 输出 数组 中 的 答案 
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if($arrayO[$al!="™"X /数组 中 的 值 不 为 空 
if($array0[$a]==$_POST[$myrowO[kt_id]]) {// 如 果 数 组 中 的 值 与 表单 提交 的 值 相等 
$str0=$_POST[$myrow0[kt_idJ]j; 。“ // 将 表单 提交 的 值 赋 给 变量 $str0 


} 
} 
3 
} 
J A 
for($a=0;$a<count($array0);$a++){ /| 执行 for 循环 语句 ， 根 据 考题 答案 来 决定 循环 的 次 数 
if($arrayO[$a]!="X // 判 断 如 果 考 题 答案 不 为 空 ， 则 执行 下 面 的 内 容 
?> 
<tr> 
<td height="20" bgcolor="#FFFFFF">&nbsp;&nbsp; 
<!-- 创 建 单 选 按钮 组 ， 用 于 提交 答案 -> 
<input type="radio" name="<?php echo $myrowO[kt_id];?>" value="<?php echo $array0[$a];?>"> 
<?php echo $array0[$a];?> 
<! 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 : > 
</td> 
<td bgcolor="#FFFFFF"> &nbsp; 
<?php 
六 wereewesttvstetewwt 江 | 断 提 交 的 答案 是 否 正确 ， 并 且 输 出 答案 tereeeereerererreesreeeoes 
if($_POST[$myrowO[kt_id]]==true}{ 1/ 判断 表单 提交 的 值 是 否 为 空 
if($a==0) // 如 果 变 量 $a==0， 则 输出 下 面 的 内 容 
if($myrowO["kt_zqdaan"]==$strOX{ // 判 断 提交 的 答案 是 否 与 数据 库 中 的 正确 答案 相同 
// 如 果 提 交 的 答案 正确 ， 则 输出 下 面 的 内 容 
echo "您 输入 的 答案 &nbsp;"; 
echo "<font color='#FF0000'>".substr($str0,0,1)."</font>"; // 获 取 答 案 的 选项 
echo "&nbsp; 正 确 &nbsp;&nbsp; 分 数 :"; 
echo "<font color='#FF0000'>".$myrowO[kt_fs]."</font>"; // 输 出 考题 的 分 数 
S$fen0+=$myrowO["kt_fs"]; // 对 考题 的 分 数 进行 核算 
jelse{ /如果 提交 的 答案 不 正确 ， 则 输出 下 面 的 内 容 
echo "您 输入 的 答案 &nbsp;"; 
echo "<font color=#FF0000'>".substr($str0,0,1)."</font>"; 
echo "&nbsp; 错 误 &nbsp;&nbsp;"; 
echo "正确 答案 :&nbsp;<font color=#FF0000'>".substr($myrowO[kt_zqdaan],0,1)."</font>" ; 
} 
} 
?></td> 
<?php 
} 
} $x++; // 实 现 考题 题 号 的 循环 
a 


4.7.3 应 用 Ajax 在 线 答题 的 实现 过 程 


国 ” 在线 答 题 使 用 的 数据 表 : tb_kt 
在 线 答题 是 本 项 目 中 最 核心 的 技术 ， 也 是 本 系统 开发 的 最 终 目 的 。 在 线 答题 模块 的 主要 功能 是 实 
现在 线 考试 ， 在 当前 页 面 中 输出 考题 答案 和 考试 成 绩 ， 并 且 采 用 计时 控制 考试 时 间 ， 如 果 考 试 超过 考 


他 
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试 时 间 ， 则 系统 自动 提交 考题 答案 ， 并 给 出 考试 成 绩 。 

考试 时 间 计时 和 剩余 时 间 的 计算 是 应 用 Ajax 技术 来 实现 的 ， 有 关 该 技术 的 实现 方法 可 以 参考 4.10.2 
节 ， 这 里 不 再 讲解 。 

在 线 答题 技术 中 对 单 选 题 的 处 理 方法 已 经 在 技术 分 析 中 进行 了 详细 的 讲解 ， 这 里 就 不 再 袭 述 ， 下 
面 讲解 在 线 答题 中 多 选 题 的 处 理 方法 ， 其 实现 的 原理 与 单 选 题 是 相同 的 ， 只 是 在 输出 考题 的 答案 时 使 
用 的 是 复 选 框 ， 不 再 是 单 选 按钮 组 ， 并 且 在 对 复 选 框 名 称 进 行 设置 时 增加 一 个 变量 来 对 不 同 考题 的 答 
案 进 行 区 分 。 首 先 以 上 一 页 表单 中 提交 的 值 为 条 件 ， 执行 查询 语句 ， 通 过 while 语句 输出 查询 结果 ; 然 
后 根据 输出 的 查询 结果 创建 复 选 框 ， 生 成 考题 ， 最 后 ， 将 考题 答案 提交 到 本 页 ， 对 提交 的 答案 与 数据 
库 中 存储 的 正确 答案 进行 比较 ， 并 且 将 答案 输出 到 本 页 中 。 

首先 从 数据 库 中 读 取 数据 ， 定 义 输出 考题 时 应 用 到 的 变量 值 ， 并 且 输 出 考题 的 编号 和 内 容 。 关 键 
代码 如 下 : 

倒 性 08 ”代码 位 置 ， 光盘 \TM\04\online_ks\ksks.php 

<?phl 
根据 表单 提交 的 值 执行 查询 语句 ， 从 数据 库 中 读 取 符合 条 件 的 数据 

Squery1=mssql_query("select * from tb_kt where kt_Ilb='$kt_lbes' and kt_Ilx="1' and kt_small_Ilb='$kt_ 


small_Ib"); 
$y=1; // 定 义 变量 $y 用 于 生成 考题 编号 
$fen1=0; // 定 义 变量 $fen1 用 于 分 数 的 统计 
Sii=0; // 定 义 变量 $ii 用 于 控制 考题 答案 输出 的 次 数 
while($myrow1=mssql_fetch_array($query1)){ // 循 环 输出 数据 
?> 
< 一 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 输出 考题 编号 和 考题 的 内 容 - 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 > 
<tr> 


<td height="20" bgcolor="#FFFFFF">&nbsp;&nbsp;<?php echo $y.".".$myrow1["kt_nr"]?> </td> 
<td bgcolor="#FFFFFF"><?php echo $myrow1[kt_fs];?></td> 
</tr> 
一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一- > 


然后 ， 对 表单 中 提交 的 答案 进行 处 理 ， 与 数据 库 中 存储 的 正确 答案 相 比 较 ， 如 果 答 案 正 确 ， 则 将 
表单 提交 的 答案 写 入 到 一 个 空 数组 中 。 代 码 如 下 : 
倒 程 09 ”代码 位 置 ， 光盘 \TM\04\online_ks\ksks.php 


<?php 
reresawesssssesets 对 考题 提交 的 答案 进行 处 理 ， 输 出 考题 答案 sseesessesesossseess/ 
$array_a1=array(); /定义 一 个 空 数 组 


/将 数据 库 中 考题 的 答案 以 * 进 行 分 割 ， 存 储 到 数组 $array1 中 
© $aray1=explode("™*",$myrow1["kt_daan"]); 


if($_POST[Submit]!=™"){ 1/ 判断 表单 提交 的 值 是 否 为 空 
for($i=0;$i<count($array1);$i++X{ // 循 环 读 取 数 组 $array1 中 的 考题 答案 
if($array1[Si]!="){ // 判 断 数组 中 的 答案 是 否 为 空 


// 对 表单 中 提交 的 答案 与 数据 库 中 的 答案 进行 比较 
if($array1[$]==$_POST[$myrow1[kt_id]."-".$i){ 
// 如 果 答案 相同 ， 则 将 表单 提交 的 答案 写 入 到 空 数组 $array_a1 中 
@ array_push($array_a1,$_POST[$myrow1[kt_id]."-".$i]); 
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< 代码 贴 圭 
@ explode() 函 数 : 将 字符 囊 依 指定 的 字符 囊 或 字符 separator 切 开 。 语法 如 下 : 


array explode(string separator, string string, [int limit]) 


该 函数 返回 由 字符 串 组 成 的 数组 ， 每 个 元 素 都 是 string 的 一 个 子囊 ， 它 们 被 字符 事 separator 作为 边界 点 分 隔 出 来 。 
如 果 设 置 了 limit 参数 ， 则 返回 的 数组 包含 最 多 limit 个 元 素 ， 而 最 后 那个 元 素 将 包含 string 的 剩余 部 分 。 如 果 separator 
为 空 字符 串 (" )，explode() 函 数 将 返回 false; 如 果 separator 所 包含 的 值 在 string 中 找 不 到 ， 那 么 explode() 函 数 将 返回 
包含 string 单个 元 素 的 数组 。 如 果 参 数 limit 是 负数 ， 则 返回 除了 最 后 的 -limit 个 元 素 外 的 所 有 元 素 。 

@ array_push() 函 数 : 将 数组 当成 一 个 栈 ， 并 将 传 入 的 变量 压 入 该 数组 的 末尾 。 该 数组 的 长 度 将 增加 入 栈 变量 的 数 
目 。 返 回 数组 新 的 单元 总 数 。 语 法 如 下 : 

int array_push ( array array, mixed var [, mixed ...]) 

参数 array 指 输 入 的 数组 ; 参数 var 是 必要 参数 ， 用 来 压 入 数组 。 


接着 创建 考题 提交 的 复 选 框 ， 根 据 考题 的 ID 和 变量 $I 定义 复 选 框 的 名 称 。 代 码 如 下 : 
倒 程 10 ”代码 位 置 ， 光盘 \TM\04\online_ks\ksks.php 


<?php 
$str1=implode("™*",$array_a1); /以 * 来 分 割 数组 $array_a1 中 的 数据 ， 返 回 值 为 变量 $str1 
for($i=0;$i<count($array1);$i++X{ // 根 据 数据 库 中 存储 的 答案 执行 循环 语句 
if($array1[$i]!=""){ // 判 断 答案 不 为 空 
?> 
<tr><td height="20" bgcolor="#FFFFFF">&nbsp;&nbsp; 
< 上 -- -一 -一 一- 一- 一 -创建 答案 提交 的 复 选 框 ， 并 且 设置 复 选 框 的 名 称 为 考题 的 id 和 变量 $i 一 一- 一--- 一 > 
<input type='checkbox' name='<?php echo $myrow1[kt id]."-".$i?>' value='<?php echo $array1[$i];?>'> 
<?php echo $array1[$ij];?> 
<! 


ee < 


</td><td bgcolor="#FFFFFF">&nbsp; 


再 次 对 提交 的 答案 进行 判断 ， 其 中 主要 应 用 的 是 explode0、1list0 和 substr(O) 函 数 来 实现 答案 的 判断 
和 和 输 出。 程序 代码 如 下 : 
倒 程 11 ”代码 位 置 ， 光盘 \TM\04\online_ks\ksks.php 
<?php 
]* xexxeseesxxexssxsssx 对 提交 的 答案 进行 判断 ， 并 且 输 出 答案 和 最 终 得 分 :eeexeeeeseesesexeeeeeenl 
if($_POST[$myrow1[kt_id]."-".$i]==true}{ // 判 断 提交 的 答案 是 否 为 真 
if($ii==0X{ // 判 断 变量 $ii 的 值 是 否 为 0， 控 制 答案 输出 的 次 数 
if($myrow1["kt_zqdaan"]==$str1X ”// 判 断 表 单 提交 的 答案 是 否 与 数据 库 中 的 正确 答案 相等 
六 sr 如果 答案 正确 ， 则 输出 提交 的 答案 ， 并 且 给 出 该 题 的 得 分 从 sse 让 


echo "您 输入 的 答案 &nbsp;"; 

$arrayesg=explode(™",$str1); // 以 * 来 分 隔 提交 的 答案 
[9 for($gg=0;$gg<count($arrayesg);$gg++}{ /循环 读 取 答案 的 内 容 
@ list($name,$valuesg)=each($arrayesg); /| 输出 答案 


echo "<font color=#FF0000>".substr($valuesg,0,1)."</font>";， /截取 答 案 的 内 容 
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echo "&nbsp; 正 确 &nbsp;&nbsp; 分 数 :"; 
echo "<font color='#FF0000'>".$myrow1[kt_fs]."</font>"; 


Sfen1+=$myrow1["kt_fs"]; // 统 计 考题 的 分 数 
Jelse{ 
广 *ee**xx*** 如 果 输 出 的 答案 错误 ， 则 提示 您 输入 的 答案 错误 ， 并 给 出 正确 的 答案 ********** */ 
echo "您 输入 的 答案 &nbsp;"; 
S$arrayes=explode(™*",$str1); 
for($g=0;$g<count($arrayes);$g++){ 
list($name, $values)=each($arrayes); 
echo "<font color=#FF0000'>".substr($values,0,1)."</font>"; 
} 
echo "&nbsp; 错 误 &nbsp;&nbsp; 正 确 答案 :&nbsp;"; 
S$arrayes=explode("™*",$myrow1[kt_zqdaan]); 
for($g=0;$g<count($arrayes);$g++){ 
list($name, $values)=each($arrayes); 
echo "<font color=#FF0000'>".substr($values,0,1)."</font>"; 
}Sii=1; // 控 制 每 道 题 的 答案 只 输出 一 次 数 
} 
?></td></tr> 
<?php} } 
By++; // 控 制 每 道 题 的 题 号 
} 
?> 
< 代码 贴 寺 


@ count() 函 数 : 返回 数组 中 的 单元 数目 ， 用 来 计算 数组 中 值 的 个 数 。 语 法 如 下 : 


int count( mixed array [, int mode]) 


参数 array 是 必要 参数 ， 指 输入 的 数组 ; 参数 mode 是 可 选 参 数 ， 默 认 值 是 0。 
@ list() 函 数 : 把 数组 中 的 值 赋 给 一 些 变量 。 与 array() 函 数 类 似 ， 这 不 是 真正 的 函数 ， 而 是 语言 结构 。list() 函 数 仅 能 
用 于 数字 索引 的 数组 ， 且 数字 索引 从 0 开始 。 语 法 如 下 : 


void list( mixed .…) 


参数 mixed 为 被 赋值 的 变量 名 称 。 
each(O) 函 数 用 于 返回 数组 中 当前 指针 位 置 的 键 名 和 对 应 的 值 ， 并 向 前 移动 数组 指针 。 


array each( array array) 


参数 array 为 输入 的 数组 。 
4.7.4 分数 统计 和 成 绩 保存 的 实现 过 程 


国 分 数 统计 和 成 绩 保存 使 用 的 数据 表 : tb_user 
在 线 考 试 模块 中 不 但 完成 考试 的 答题 和 判 卷 的 操作 ， 而 且 可 以 直接 对 考试 的 分 数 进 行 统计 ， 获 取 
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考生 的 最 终 得 分 ， 将 考生 的 成 绩 保 存 到 数据 库 中 ， 并 且 更 改 数据 库 中 考生 的 考试 信息 ， 说 明 该 考生 已 
经 完成 本 次 考试 ， 不 可 以 再 进行 本 类 别 的 考试 。 考 试 分 数 统计 和 存储 考试 成 绩 的 关键 代码 如 下 : 
倒 程 12 ”代码 位 置 ， 光盘 \TM\04\online ks\ksks.php 


<?php 
eeeeeeexxxxx* 统 计 考生 的 考试 成 绩 ， 其 中 的 变量 分 别 代表 考生 各 个 题 型 的 得 分 "ee*e** */ 
$zf=$fenO0+$fen1+$fen2+$fen3; // 考 试题 分 数 统计 
echo "您 的 总 成 绩 是 :"; // 输 出 字符 串 
echo $zf; /输出 考试 成 绩 
?> 
<?php 
so 将 考 生 的 成绩 存储 到 数据 库 吉 站 全 全 全 ee 全 2 二 抽 
if($Submit==" 提 交 ")f /首先 判断 考生 是 否 提交 考题 的 答案 
$data=date("Y-m-d H:i:s"); /获取 考试 的 时 间 


$grade="update tb_user set online_grade='$zf ,online_subject='$_POST[kt_ lbes],online_pt="1',online_date= '$data' 
where online_number='$_SESSION[online_numben"; 

$grade_result=mssql_query($grade); 

} 


?> 


4.7.5 ”单元 测试 


在 完成 在 线 考试 模块 的 设计 后 对 该 模块 进行 测试 ， 发 现在 执行 多 选 题 提交 的 过 程 中 ， 没 有 任何 值 
返回 。 但 是 在 执行 单 选 题 提交 时 可 以 正常 运行 ， 查 看 该 文件 的 源 文件 发 现 ， 原 来 错误 出 现在 复 选 框 名 
称 的 设置 上 ， 所 有 复 选 框 都 是 相同 的 值 ， 其 运行 结果 如 图 4.23 所 示 。 


x 11] 一 记事 本 TE 
Ei Tr TY 


tr> 
te height-"20" bgcolor-" WFTFTFr">nbsps nbsp; 
Cinput type="checkbox" nane="17° value=’C PHP'》C PHPC/td> 
tt bgcoler-"sFFFFFF">enbspi</tdy 
ter? 


机 ether20 bgcolor FFFFFF -Yendep stnbep; 
cnput type="checkbox” nane="17° value=*D asP*> D asP<C/to> 
td bgeoler= "FFFFFF">anbspi</tdy 
ry 


图 4.23 查看 源 文件 的 结果 


在 设置 复 选 框 的 名 称 时 必须 为 每 一 个 题 设置 不 同 的 复 选 框 名 称 ， 否 则 将 不 能 获取 到 表单 提交 的 数 
据 ， 这 就 是 多 选 题 与 单 选 题 进行 输出 和 判断 的 主要 区 别 ， 多 选 题 中 有 多 个 答案 ， 而 单 选 题 只 有 一 个 ， 
所 以 单 选 题 的 单 选 按钮 组 的 名 称 可 以 直接 使 用 考题 的 ID 进行 命名 ， 但 是 多 选 题 必须 在 考题 ID 的 基础 
之 上 再 定义 一 个 变量 来 区 分 每 个 复 选 框 的 名 称 。 在 程序 中 ， 错 误 的 复 选 框 名 称 定义 的 代码 如 下 : 


<input type="radio" name="<?php echo $myrowO[kt_id];?>" value="<?php echo $array0[$a];?>"> 
正确 的 复 选 框 名 称 定义 应 该 是 在 考题 ID 的 基础 之 上 加 一 个 变量 $i， 程 序 代 码 如 下 : 
<input type='checkbox' name='<?php echo $myrow1[kt_id]."-".$i?>' value='<?php echo $array1[$i];?>'> 


对 程序 代码 进行 修改 ， 重 新 保存 ， 程 序 可 以 正常 运行 。 
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4.8 后 台 首 


半 | 
尖 
rs 


4.8.1 后 台 首 页 概述 


网 络 在 线 考试 系统 的 后 台 首页 是 管理 员 对 网 站 信息 进行 管理 的 首页 面 。 在 该 页 面 中 ， 管 理 员 可 以 
清楚 地 了 解 网 站 后 台 管 理 系 统 包含 的 基本 操作 。 网 络 在 线 考试 网 后 台 首 页 包含 的 主要 模块 介绍 如 下 。 

回 ”管理 员 信息 管理 : 主要 用 于 修改 管理 员 信息 。 

回 ”考生 信息 管理 : 主要 包括 查看 注册 考生 信息 列表 和 考生 信息 查询 、 考 试 成 绩 查 询 、 删 除 已 注 
册 的 考生 信息 。 
考题 类 别管 理 : 主要 包括 查看 考题 类 别 列表 、 添 加 考题 类 别 信息 和 删除 考题 类 别 信息 。 
考题 信息 添加 : 主要 用 于 为 各 类 套 题 添加 单 选 题 、 多 选 题 、 问 答题 和 论述 题 ， 并 设置 每 题 的 
分 数 及 内 容 。 

加 ”考题 信息 管理 ， 主 要 包括 查看 考题 类 别 列表 、 修 改 套 题 信 息 和 删除 套 题 信息 。 

回 ”退出 管理 : 主要 用 于 退出 后 台 管 理 系统 。 

下 面 看 一 下 本 案例 中 提供 的 后 台 首页 , 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\04\online_ks\admin\index. 
php， 如 图 4.24 所 示 。 


娘 数 | 专题 和 列 


100 | Pi 上 坝 才 是 | 
ET 全 | 7 本 上 山大 


图 424 ”网络 在 线 考试 系统 的 后 台 首 页 
4.8.2 后台 首 页 技术 分 析 


网 络 在 线 考试 后 台 主 要 应 用 switch 语句 实现 了 一 种 类 似 于 框架 的 网 页 嵌 套 技术 。 为 了 能 够 更 好 地 
理解 这 个 技术 ， 下 面 进行 详细 介绍 。 

switch 语句 一 行 接 一行 地 执行 《实际 上 是 语句 接 语句 )， 开 始 时 没有 代码 被 执行 。 仅 当 一 个 case 语 
句 中 的 值 和 switch 表达 式 的 值 匹 配 时 ，PHP 才 开 始 执行 语句 ， 直 到 switch 的 程序 段 结束 或 者 遇 到 第 一 
个 break 语句 为 止 。 如 果 不 在 case 的 语句 段 最 后 写 上 break，PHP 将 继续 执行 下 一 个 case 中 的 语句 段 。 


_- 国 
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switch 语句 的 语法 如 下 : 


switch(expr) { 
case value : statement; break; 


default : statement; break:; 


} 
switch 语句 的 参数 说 明 如 表 4.7 所 示 。 
表 4.7 ”switch 语句 的 参数 说 明 


参数 说 有明 
expr 必要 参数 。 定 义 判 断 的 条 件 表达 式 
statement 必要 参数 。 满 足 条 件 后 ， 循 环 执行 的 语句 


4.8.3 ”后台 首页 的 实现 过 程 


网 络 在 线 考 试 后 台 主 要 应 用 switch 语句 和 include 包含 语句 实现 了 类 似 于 框架 的 网 页 典 套 技术 。 其 
中 ， 应 用 switch 语句 ， 根 据 变量 标识 $Shtgl 提交 的 值 进行 判断 ， 应 用 include 包含 语句 调用 不 同 功能 模 
块 的 脚本 文件 。 代 码 如 下 : 

倒 程 13 ”代码 位 置 ， 光盘 \TM\04\online_ks\admin\index.php 


<?php 
switch($htgl){ // 判 断后 台 超 链 接 名 称 
case "考生 信息 管理 ": // 如 果 名 称 为 “考生 信息 管理 ” 
include("ksxx_gl.php"); // 则 应 用 include() 语 句 调用 ksxx_gl.php 文件 
break; // 退 出 语句 段 的 执行 
case "考题 类 别管 理 ": // 如 果 名 称 为 “考题 类 别管 理 ” 
include("ktlb_gl.php"): // 则 应 用 include() 语 句 调用 ktlb_gl.php 文件 
break; // 退 出 语句 段 的 执行 
case "考题 信息 添加 ": // 如 果 名 称 为 “考题 信息 添加 ” 
include("ktxx_tj.php"); // 则 应 用 include() 语 句 调用 ktxx_tj.php 文件 
break; // 退 出 语句 段 的 执行 
case "考题 信息 管理 ": // 如 果 名 称 为 “考题 信息 管理 ” 
include("ktxx_gl.php"); // 则 应 用 include() 语 句 调用 ktxx_gl.php 文件 
break; // 退 出 语句 段 的 执行 


4.9 考题 信息 管理 模块 设计 


4.9.1 考题 信息 管理 模块 的 概述 


考题 信息 管理 模块 主要 包括 查询 考题 信息 、 添 加 考题 信息 、 修 改 考题 信息 和 删除 考题 信息 等 4 个 
功能 。 考 题 信息 管理 模块 的 框架 如 图 4.25 所 示 。 


@ 
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查询 考题 信息 | | | | 添加 考题 信息 | | [修改 考题 信息 | | [ 【删除 考 愿 信息 | | 
图 4.25 考题 信息 管理 模块 的 框架 图 


4.9.2 考题 信息 管理 模块 的 技术 分 析 


在 实现 考题 信息 管理 模块 时 ， 为 了 更 好 地 管理 ， 因 此 把 考题 类 别 单独 存储 于 一 个 数据 表 tb_lb 中 ， 
这 样 在 录入 考题 信息 时 就 可 以 把 考题 类 别 以 下 拉 列 表 的 形式 从 数据 库 中 读 取 出 来 。 这 种 从 下 拉 列 表 中 
动态 显示 数据 表 某 列 的 字段 值 的 方法 ， 不 仅 方便 管理 员 更 快捷 、 灵 活 地 操作 网 络 在 线 考 试 系统 ， 更 大 
大 提高 了 工作 效率 ， 达 到 事半功倍 的 效果 。 

下 拉 列 表 是 一 种 最 节省 空间 的 数据 显示 方式 ， 正 常 状态 下 只 能 看 到 一 个 选项 ， 单 击 控制 按钮 后 ， 
可 以 显示 一 定数 量 的 选项 ， 如 果 超 出 这 个 数量 ， 会 自动 显示 滚动 条 ， 管 理 员 可 以 通过 拖 动 滚动 条 来 选 
择 各 选项 。 

下 面 介绍 从 下 拉 列 表 中 动态 显示 数据 表 某 列 的 字段 值 的 方法 。 

首页 ， 创 建 与 数据 库 的 连接 。 代 码 如 下 : 

<?php 

$id=mssql_connect(localhost','sa',”"); 

mssql_select_db("db_online ",$id); 

?> 


然后 应 用 下 拉 列 表 框 和 select 查询 语句 相 结合 实现 在 下 拉 列 表 中 显示 数据 表 中 的 online_ktlb 字段 
的 值 ， 通 过 while 循环 语句 进行 输出 。 代 码 如 下 : 


<?php 
$query=mssql_query("select * from tb_ktlb"); // 查 询 考题 类 别 表 中 的 信息 
while($myrow=mssql_fetch_array($query)X{ // 应 用 while 循环 语句 输出 考题 信息 
?> 
< 一 一 一 一 一 一 一 榆 出 考古 类 别 表 中 的 online_ktb 字段 -一 一 一 一 一 一 一 一 一 一 一 一 一 -> 
<option value="<?php echo $myrow[online_ktlb];?>"><?php echo $myrow[online_ktlb];?></option> 
<?php 
/while 循环 语句 结束 
?> 


4.9.3 ”考题 信息 添加 的 实现 过 程 
国 ” 添 加 考试 题目 信息 使 用 的 数据 表 : tb_ Lesson、tb_taoTi 和 tb_Questions 


考试 题目 添加 包含 两 个 步骤 : 一 是 为 添加 的 考试 题目 选择 专业 、 课 程 以 及 套 题 ， 二 是 将 填写 的 考 
试题 目 信息 插入 到 数据 库 中 。 考 题 信息 添加 页 面 的 运行 结果 如 图 4.26 所 示 。 
® 
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图 4.26 考题 信息 添加 页 面 的 运行 结果 
考题 信息 添加 页 面 涉及 的 HTML 表单 的 重要 元 素 如 表 4.8 所 示 。 
表 4.8 考题 信息 添加 页 面 涉及 的 HTML 表单 的 重要 元 素 


名 称 类 型 重要 属性 含义 
form2 form method="post" action="ktxx_tj_ok.php” 表单 
<?php 
$query=mssql_query("select * from tb_ktlb"); 
while($myrow=mssql_fetch_array($query)){ 
kt_lb Select _ 考题 类 别 
<option value="<?php echo $myrow[online_ktlb];?> "> 
<?php echo $myrow[online_ktlb];?> 
</option> 
<?php } ?> 
<select> 
<option value=" 第 一 套 题 "> 第 一 套 题 </option> 
<option value=" 第 二 套 题 "> 第 二 套 题 <option> 
<option value=" 第 三 套 题 "> 第 三 套 题 <option> 
<option value=" 第 四 套 题 "> 第 四 套 题 </option> 


</select> 


kt_small_lb select 所 属 套 题 


<select> 
<option value="2"> 简 答 </option> 
<option value="3"> 论 述 </option> 
kt_lx lect 
| ee <option value="0"> 单 选 </option> 考题 类 型 


<option value="1"> 多 选 </option> 


</select> 
kt fs text size="10" 考试 成 绩 
kt nr textarea cols="60" rows="5" 考题 内 容 
kt_daan textarea cols="60" rows="5" 考题 答案 
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续 表 


名 称 类 型 
kt_zqdaan | textarea 
Submit2 


重要 属性 
cols="60" rows="5" | 考题 正确 答案 
value=" 提 交 考 题 " 


submit 


添加 考试 题目 首先 要 选择 考题 的 类 别 ， 然 后 选择 所 属 套 题 ， 再 选择 考题 类 型 ， 其 实现 过 程 在 表 4.8 
所 示 表单 的 重要 属性 中 给 出 了 完整 的 代码 ， 这 里 不 再 袭 述 ， 最 后 根据 考题 类 别 将 考题 内 容 添加 完整 。 

提交 考题 表单 到 数据 处 理 页 ， 程 序 处 理 页 面 首先 应 用 变量 获取 到 表单 数据 ， 然 后 应 用 insert…into 
语句 将 其 插入 到 考题 信息 表 tb_kt 中 。 如 果 考 题 信息 添加 成 功 ， 则 弹出 提示 信息 ， 并 重新 定位 到 考题 信 
息 添 加 页 面 。 数 据 处 理 页 的 完整 代码 如 下 : 

全 性 14 ”代码 位 置 ， 光盘 \TM\04\online_ks\admin\ktxx_tj_ok.php 


<?php 
include("../conn/conn.php"); // 连 接 数据 库 文件 
if($Submit2==" 提 交 考 题 "){ // 如 果 单 击 了 “提交 考题 ”按钮 ， 则 执行 下 面 语句 


$queryes=mssql_query("insert into tb_kt 


aan','$kt_small_Ib")"); // 向 考题 信息 表 中 添加 数据 
if($queryes){ // 如 果 添 加 操作 成 功 ， 则 弹出 提示 ， 并 重 定向 网 页 
echo "<script>alert(' 考 题 添 加 成 功 ! '); window.location.href='index.php?htgl= 考 题 信息 添 加 ';</script>"; 
} 
i 
9% 


4.9.4 查询 考题 信息 的 实现 过 程 


国 查看 考试 题目 列表 使 用 的 数据 表 : tb kt、tb_ktlb 

管理 员 登 录 后 ， 单 击 “ 考 题 信息 管理 ” 超 链接 ， 进 入 到 查询 考题 信息 页 面 ， 选 择 考题 类 别 后 ， 单 
击 “ 考 题 查找 ”按钮 ， 将 查询 出 该 类 别 下 的 所 有 考题 信息 。 同 时 提供 修改 考题 信息 和 删除 考题 信息 的 
功能 。 查 询 考题 信息 页 面 的 运行 结果 如 图 4.27 所 示 。 


| sy [5 
ER 


4.27 查询 考题 信息 页 面 的 运行 结果 
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查询 考题 信息 页 面 涉及 的 HTML 表单 的 重要 元 素 如 表 4.9 所 示 。 


表 4.9 查询 考题 信息 页 面 涉 及 的 HTML 表单 的 重要 元 素 
名 称 类 型 重要 属性 含义 

forml form method="post" action="index.php?htgl= 考 题 信息 管理 " 查询 考题 表单 

<selectname= "kt lb" id="kt lb"> 

<?php 

Squery=mssql_query("select * from tb_ktlb"); 

while($myrow=mssql_fetch_array($query)){ 
kt_lb select S 考题 类 别 

<option value="<?php echo $myrow[online_ktlb];?>"> 

<?php echo $myrow[online_ktlb];?></option> 

<?php }7> 

</select>, 
Submit submit value=" 考 题 查找 " “考题 查找 ”按钮 
form2 form method="post" action="ktxx_gl_ok.php" 考题 信息 表单 
kt lb text value= "<?php echo Smyrow[kt_Ib]:2>” 考题 类 别 显示 
kt lx text value="<?php echo $myrow[kt Ix]:?>" 考题 类 型 
kt fs text value="<?php echo Smyrow[kt fs]:?>" 考试 成 绩 
kt id hidden value=" <?php echo Smyrow[kt idl?>" 考题 类 别 id 
Submit2 submit value=" 修 改 " “修改 ”按钮 
Submit3 submit value=" 删 除 " “删除 ”按钮 
kt nr textarea <?php echo Smyrow[kt nr]:?> 考题 内 容 
kt_daan textarea <?php echo $myrow[kt daan]:?> 考题 答案 
kt_zqdaan textarea <?php echo Smyrow[kt zqdaan]:2?> 考题 正确 答案 


在 考试 题目 查询 页 面 中 ， 首 先 建立 用 于 查询 的 表单 fprm1， 该 表单 中 包含 “考题 类 别 ” 列 表 / 菜 单 
控件 以 及 “考题 查找 ”按钮 。 

当 管理 员 选 择 考题 类 别 后 ， 单 击 “ 考 题 查找 ”按钮 ， 提 交 考 题 类别 kt_lb 到 当前 页 ， 然 后 根据 获取 
到 的 考题 类 别 检索 考题 信息 表 tb_kt， 并 将 该 类 别 下 的 所 有 考题 信息 输出 到 浏览 器 。 代 码 如 下 : 

倒 程 15 ”代码 位 置 ， 光盘 \TM\04\online_ks\admin\ktxx_gl.php 


ee 考量 信息 最 示 一 一 
<table width="682" height="168" border="0" cellpadding="0" cellspacing="1" bgcolor="#5D554A"> 
<?php 
include("../conn/conn.php"); 
@ $query=mssql_query("select* from tb_kt where kt_lb='$kt_lb"); 
© while($myrow=mssql fetch_array($query)}{ 
?> 
<form name="form2" method="post" action="ktxx_gl_ok.php"> 
<tr> 

< 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 考题 类 别 显示 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 ~ 

<td width="112" height="27" align="center" bgcolor="#DDDDDD"> 考 题 类 别 </td> 

<td width="117" align="center" bgcolor="#DDDDDD"> 

<input name="kt_lb" type="text" value="<?php echo $myrow[kt_Ib];?>" size="8"> 
</td> 


// 连 接 数 据 库 文件 
// 根 据 选择 考题 类 别 检索 数据 信息 
// 将 查询 结果 应 用 while 循环 输出 


EG 
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< 上 -一 一 考题 类 型 显示 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 > 
<td width="180" align="center" bgcolor="#DDDDDD"> 考 题 类 型 


<input name="kt_Ix" type="text" value="<?php echo $myrow[kt_Ix];?>" size="10"> 
</td> 


i i 


-一 一 一 一 考试 成 绩 显示 一 一 一 一 一 一 一 -> 
<td width="148" align="center" bgcolor="#DDDDDD"> 分 数 
<input name="kt_fs" type="text" value="<?php echo $myrow[kt_fs];?>" size="5"> 
</td> 
< 上 - 一 一 一 一 一 一 一 一 一 一 一 一 -建立 考题 ID 隐藏 域 ， 添 加 “修改 ” “删除” 按钮 一 一 -一 一 一 一 一 一 --- -> 
<td width="99" rowspan="4" align="center" bgcolor="#FFFFFF"> 
<input type="hidden" name="kt_id" value="<?php echo $myrow[kt_id]?>"> 
<input type="submit" name="Submit2" value=" 修 改 ">/<input type="submit" name="Submit3" value=" 删 除 "> 
</td> 
</tr> 
< 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 - 
<tr> 
<td height="43" align="center" bgcolor="#DDDDDD"> 考 题 内 容 </td> 
<td colspan="3" align="center bgcolor="#FFFFFF"><textarea name="kt_nr" cols="60" 
echo $myrow[kt_nr];?></textarea></td> 
</tr> 
< 考题 答案 显示 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 -> 
<tr> 
<td height="46" align="center" bgcolor="#DDDDDD"> 考 题 答 案 </td> 
<td colspan="3" align="center" bgcolor="#FFFFFF"><textarea name="kt_daan" cols="60" rows="5"><?php 
echo $myrow[kt_daan];?></textarea></td> 
</tr> 
< 考题 正确 答案 显示 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 -> 
<tr> 
<td height="33" align="center" bgcolor="#DDDDDD"> 考 题 正 确 答案 </td> 
<td colspan="3" align="center" bgcolor="#FFFFFF"><textarea name="kt_zqdaan" cols="60" rows="5"> <?php 
echo $myrow[kt_zqdaan];?></textarea></td> 
</tr> 
</form> 
<?php 
上 /while 循环 语句 结束 
?> 
</table> 


一 


rows="5"><?php 


< 代码 贴 二 
@ mssql_query() 函 数 : 向 SQL Server 服务 器 中 的 当前 活动 数据 库 发 送 一 条 SQL 查询 。 
@ mssql_fetch_array0) 函 数 : 返回 根据 从 结果 集 获 取 的 行 生 成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false。 语 法 如 下 : 


array mssql_fetch_array( resource result [, int result_type] ) 


参数 result: 资源 类 型 的 参数 ， 要 传 入 的 是 由 mssql_query0) 函 数 返 回 的 “数据 指针 ?”。 
参数 result_type: 可 选项 ， 整 数 型 参数 ， 要 传 入 的 是 MSSQL_ASSOC、MSSQL NUM 和 MSSQL BOTH 3 种 
由 PHP 定义 好 的 常数 之 一 ， 默 认 值 是 MSSQL BOTH. 
> 用 MSSQL ASSOC 只 得 到 关联 索引 。 
> 用 MSSQL NUM 只 得 到 数字 索引 。 
> 用 MSSQL_BOTH 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 。 


_ 国 
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查询 考题 信息 页 面 提供 了 修改 考题 信息 的 功能 ， 管 理 员 可 对 指定 的 考题 信息 进行 编辑 ， 单 击 “ 修 
改 ” 按 钮 后 ， 提 交 Submit2 表单 信息 到 数据 处 理 页 ktkx_glL_ok.php。 
倒 程 16 ”代码 位 置 : 光盘 \TM\04\online ks\admin\ktxx_gl_ok.php 


<?php 
include("../conn/conn.php"); 
if($Submit2==truej{ 


// 连 接 数据 库 文 件 
// 如 果 单 击 了 “修改 ”按钮 ， 则 执行 下 面 语句 


人 Daath | update 语句 修改 考题 信息 ******* 人 人 站 
$querys=mssql_query("update tb_kt set 

kt Ib='$kt_Ib',kt_Ix="$kt_Ix',kt_fs='$kt fs',kt_nr='$kt_nr,kt_daan="$kt_daan',kt_zqdaan='$kt_zqdaan' where kt_id='$kt_id"); 
人 ohotehointahahein nthe 有 果 考 题 信息 修改 成 功 ， 弹出 提示 ， 并 重 定 内 副 入 页 seeeeeweeesees6ecaseots wh 


if($querys){ 


echo "<script>alert(' 考 题 更 新 成 功 ! '); window.location.href='index.php?htgl= 考 题 信息 管理 ';</script>"; 


} 
} 


查询 考题 信息 页 面 提 供 了 删除 考题 信息 的 功能 ， 管 理 员 可 对 指定 的 考题 信息 进行 删除 ， 单 击 “ 删 
除 ” 按 钮 后 ， 提 交 Submit3 表单 信息 到 数据 处 理 页 ktxx_gl_ok.php。 
倒 程 17 代码 位 置 ， 光盘 \TM\04\online_ks\admin\ktxx_gl]_ok.php 


<?php 
include("../conn/conn.php"); 
if($Submit3==true){ 


// 连 接 数 据 库 文 件 
// 如 果 单 击 了 “删除 ”按钮 ， 则 执行 下 面 语句 


人 i delete 语句 删除 指定 的 考题 信息 **wsexewexxwexxwkxxwxhxwwkhwwh «] 
$query=mssql_query("delete from tb_kt where kt_id="$kt_id™); 
人 +kxxkkxkkkxxhhxxhkxxtx 加 果 考 是 信息 删除 成 功 ， 弹出 提示 ， 并 重 定 和 wh 


if($query)f 


echo "<script>alert(' 考 题 类 别 删 除 成 功 ! '); window.location.href='index.php?htgl= 考 题 信息 管理 ';</script>"; 


} 
} 


?> 


4.10 开发 技巧 与 难点 分 析 


4.10.1 考生 登录 编号 的 获取 


登录 编号 的 生成 主要 应 用 


的 是 mt_rand() 函 数 和 substr0 函 数 。 首 先 通过 mt_rand() 函 数 来 获取 


一 个 100000 一 999999 之 间 的 随机 数 ， 然 后 应 用 substrO) 函 数 从 该 随机 数 中 获取 6 个 数字 ， 作 为 考生 编号 。 
mt_rand() 函 数 的 功能 是 获取 随机 数 的 值 。 语 法 格式 如 下 : 


int mt_rand([int min,int max]) 


/ 
说明 如 果 mt_rand() 函 数 没有 提供 可 选 参 数 min 和 max， 则 返回 0 到 RAND_MAX 之 间 的 伪 随 


机 数 。 


& 
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下 面 应 用 mt_rand() 函 数 来 获取 100000 一 999999 之 间 的 随机 数 。 代 码 如 下 : 


<?php echo mt_rand(100000,999999); ?> 


运行 结果 为 : 321864 
4.10.2 通过 Ajax 技术 实现 计时 与 显示 剩余 时 间 


通过 Ajax 技术 实现 计时 与 显示 剩余 时 间 ， 首 先 要 创建 一 个 XMLHttpRequest 对 象 实例 ， 确 保 其 能 
够 在 所 有 支持 XMLHttpRequest 的 浏览 器 中 运行 , 将 其 代码 保存 在 一 个 名 称 为 xmlHttpRequestjs 的 文件 
中 ， 然 后 在 需要 应 用 Ajax 技术 的 页 面 中 ， 应 用 以 下 代码 包含 该 文件 。 

<script type="text/javascript" src="js/xmlHttpRequest.js"></script> 

xmlHttpRequest.js 文件 的 代码 如 下 : 

倒 性 18 ”代码 位 置 ， 光盘 \TM\04\online_ksjsxmlHttpRequest.js 


var xmlHttp = false; /创建 一 个 布尔 型 变量 , 用 来 检测 是 否 为 合法 的 IE 实例 
try{ // 检 测 是 否 使 用 的 是 IE 
xmlHttp = new ActiveXObject("Msxml2.XMLHTTP"); // 如 果 JavaScript 的 版 本 大 于 5 
} catch (e) { // 如 果 不 是 ， 则 使 用 老 版 本 的 ActiveX 对 象 
try{ // 如 果 使 用 的 是 IE 浏览 器 
xmlHttp = new ActiveXObject("Microsoft.XMLHTTP"); 
} catch (e2) 0 


} 
/eeeemmme 如 果 使 用 的 是 非 IE 浏览 器 ， 则 创建 一 个 该 对 象 的 JavaScript 实例 esseeeeeeeeers 
if (!xmlHttp && typeof XMLHttpRequest != "undefined") { 
try{ 
xmlHttp = new XMLHttpRequest(); 
j}catch(e3){ xmlHttp = false;} 
} 


接 下 来 编写 两 个 自 定义 的 JavaScript 函数 ShowTime() 和 sparetime()， 通 过 ShowTime() 函 数 读 取 显 
示 时 间 文 件 ShowTime.php 中 的 数据 ; 通过 sparetime0) 函 数 读 取 获取 剩余 时 间 文 件 sparetime.php 中 的 数 
据 。ShowTime0 和 sparetime() 函 数 的 代码 如 下 : 

倒 竹 19 ”代码 位 置 : 光盘 \TM\04\online_ksijsxmlHttpRequest.js 

<script type="text/javascript"> 


timer = window.setInterval("ShowTime()",1000); /每 隔 一 秒 钟 调用 一 次 ShowTime() 函 数 
// 定 义 ShowTime() 函 数 以 通过 XMLHttpRequest 对 象 读 取 ShowTime.php 文件 中 的 数据 
function ShowTime(){ 
xmlHttp.open("post","showtime.php", true); /以 post 方式 发 送 一 个 新 请 求 
xmlHttp.onreadystatechange = function(){ 
说 xmlHttp.readyState == 4X{ // 如 果 是 服务 器 响应 发 出 的 请 求 , 则 执行 以 下 操作 
tet = xmlHttp.responseText; // 获 取 返 回 的 响应 信息 


获取 到 的 信息 赋予 指定 的 DIV 标记 z#sssxsssxxxtxsssxs 区 
document.getElementByld("show_time").innerHTML = tet; 


全 
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xmlHttp.send(null); // 发 送 请 求 

} 

</script> 

<script type="text/javascript"> 

time = window.setInterval("sparetime()",1000); /每 隔 一 秒 钟 调用 一 次 sparetime() 函 数 


六 st 定义 Sparetime() 函 数 以 通过 XMLHttpRequest 对 象 读 取 sparetime.php 文件 中 的 数据 rss 可 
function sparetime(){ 


xmlHttp.open("post","sparetime.php", true); /以 post 方式 发 送 一 个 新 请 求 
xmlHttp.onreadystatechange = function(){ 
if(xmlHttp.readyState == 4){ // 如 果 是 服务 器 响应 发 出 的 请 求 , 则 执行 以 下 操作 
tet = xmlHttp.response Text; /获取 返回 的 响应 信息 


A Witnienimisnenie es 生计 的 信 息 赋予 指定 的 DIV ee 让 
document.getElementByld("sparetime ").innerHTML = tet; 


iftet=="00:00"){ /判断 当 剩 余 时 间 为 00:00 时 
form1.submit(); /提交 form1 表单 中 的 数据 
} 
ee /发 送 请 求 
i 


在 ShowTime.php 文件 中 实现 当前 时 间 的 显示 。 实 现 的 原理 是 ， 获取 一 个 在 session 变量 中 存储 的 
考试 开始 时 间 的 时 间 戳 ， 然 后 再 应 用 mktime0 函 数 获 取 当 前 时 间 的 时 间 戳 ， 应 用 当前 时 间 戳 减 去 考试 
开始 时 间 的 时 间 戳 ， 最 后 应 用 date(0) 函 数 输出 获取 的 新 时 间 戳 的 时 间 值 。 代 码 如 下 : 

倒 程 20 ”代码 位 置 ， 光 盘 \TM\04\online_ks\ShowTime.php 


<?php session_start(); /| 初始 化 session 变量 
header('Content-Type: text/xml); 
$dates=$_SESSION[dates]; /获取 session 变量 中 存储 的 考试 开始 时 间 的 时 间 戳 
$dates2=mktime(); /获取 系 统 当前 时 间 的 时 间 戳 
$dates3=$dates2-$dates; /应 用 当前 时 间 的 时 间 戳 减 去 考试 开始 时 间 的 时 间 戳 
echo date("i:s",$dates3); // 输 出 得 出 的 时 间 
?> 


在 sparetime.php 文件 中 获取 考试 的 剩余 时 间 。 实 现 的 原理 是 : 首先 设置 考试 时 间 为 20 分 钟 ， 在 考 
试 开始 时 间 的 基础 上 加 上 20 分 钟 , 然后 用 考试 时 间 减 去 系统 的 当前 时 间 , 获取 的 就 是 考试 的 剩余 时 间 ， 
当 剩余 时 间 为 00:00 时 表示 本 次 考试 结束 ， 将 考题 自动 提交 。sparetime.php 文件 的 代码 如 下 : 

倒 避 21 ”代码 位 置 ， 光盘 \TM\04\online_ks\sparetime.php 


<?php session_start(); /| 初始 化 session 变量 
header('Content-Type: text/xml); 
$dates=$_SESSION[dates]; /获取 session 变量 中 存储 的 考试 开始 时 间 的 时 间 戳 
$dates1=$dates+1*60; // 设 定 考试 时 间 
$dates2=mktime(); /获取 系统 的 当前 时 间 戳 
$dates3=$dates1-$dates2; // 计 算 考 试 的 剩余 时 间 
echo date("i:s",$dates3); // 输 出 考试 的 剩余 时 间 

?> 


& 
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最 后 在 考试 操作 页 面 中 通过 DIV 标签 来 输出 考试 时 间 和 剩余 时 间 。 代 码 如 下 : 


<div id="sparetime"></div> <!-- 输 出 考试 剩余 时 间 --> 
<div id="show_time"></div> <!-- 输 出 考试 时 间 --> 


4.11 Ajax 无 刷新 技术 专题 


4.11.1 Ajax 概述 


Ajax 是 Asynchronous JavaScript and XML 的 缩写 ， 意思 是 异步 的 JavaScript 与 XML。 

Ajax 并 不 是 一 种 新 技术 ， 或 者 说 它 不 是 一 种 技术 ， 实 际 上 ， 它 是 将 JavaScript、XHTML 和 CSS、 
DOM、XML 和 XSTL、XMLHttpRequest 等 编程 技术 以 新 的 强大 方式 组 合 而 成 ， 可 以 让 开发 人 员 构 建 
基于 PHP 技术 的 Web 应 用 ， 并 打破 了 使 用 页 面 重 载 的 惯例 。Ajax 包含 : 

XHTML 和 CSS 技术 实现 标准 页 面 。 

Document Object Model 技术 实现 动态 显示 和 交互 。 
XML 和 XSLT 技术 实现 数据 的 交换 和 维护 。 
XMLHttpRequest 技术 实现 异步 数据 接收 。 
JavaScript 绑 定 和 处 理 所 有 数据 。 

Ajax 是 一 种 运用 浏览 器 的 技术 ， 它 可 以 在 浏览 器 和 服务 器 之 间 得 到 异步 通信 机 制 进行 数据 通信 ， 
从 而 允许 浏览 器 向 服务 器 获取 少量 信息 而 不 是 刷新 整个 页 面 。 


4.11.2 Ajax 的 优点 


Ajax 是 使 用 客户 端 脚本 与 Web 服务 器 交换 数据 的 Web 应 用 开发 方法 。 这 样 ，Web 页 面 不 用 打 断 
交互 流程 进行 重新 加 载 ， 就 可 以 动态 地 更 新 。Ajax 的 优点 介绍 如 下 : 

(1) 减轻 服务 器 的 负担 。 

Ajax 的 原则 是 “ 按 需 取 数 据 ” 可 以 最 大 程度 地 减少 元 余 请 求 ， 从 而 减轻 对 服务 器 造成 的 负担 。 

(2) 无 刷新 更 新 页 面 ， 减 少 用 户 心 理 和 实际 的 等 待 时 间 。 

“ 按 需 取 数据 ”的 模式 减少 了 数据 的 实际 读 取 量 。 如 果 说 重 载 的 方式 是 从 一 个 终点 回 到 原点 再 到 另 
一 个 终点 的 话 〈 如 图 4.28 所 示 )， 那 么 Ajax 就 是 以 一 个 终点 为 基点 到 达 另 一 个 终点 〈 如 图 4.29 所 示 )。 


办 办 办 办 加 


@) CD TT 
多 @) © 


4.28 ” 重 载 方式 4.29 Ajax 方式 


其 次 ， 即 使 要 读 取 较 大 的 数据 ， 也 不 会 出 现 白 屏 的 情况 。Ajax 使 用 XMLHTTP 对 象 发 送 请 求 并 得 
到 服务 器 响应 ， 在 不 重新 载 入 整个 页 面 的 情况 下 用 JavaScript 操作 DOM 最 终 更 新 页 面 ， 所 以 在 读 取 数 
据 的 过 程 中 ， 用 户 所 面 对 的 不 是 白 屏 ， 而 是 原来 的 页 面 状态 ， 页 面具 有 接收 到 全 部 数据 后 才 更 新 相应 
部 分 的 内 容 ， 而 这 种 更 新 也 是 瞬间 的 ， 用 户 几 乎 感觉 不 到 。 
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(3) 带 来 更 好 的 用 户 体 验 。 

(4) 把 部 分 服务 器 负担 的 工作 转交 给 客户 端 ， 利 用 客户 端 闲置 的 能 力 来 处 理 任务 ， 从 而 减轻 服务 
器 和 带宽 的 负担 ， 节 约 空间 和 宽带 租用 成 本 。 

(5) 可 以 调用 外 部 数据 。 

(6) 是 一 种 基于 标准 化 并 被 广泛 支持 的 技术 ， 不 需要 下 载 插件 或 者 小 程序 。 

(7) 进一步 促进 Web 页 面 展现 形式 与 数据 的 分 离 。 


4.11.3 Ajax 的 工作 原理 


传统 的 Web 模式 强制 用 户 进入 “提交 一 等 待 一 重新 显示 ”网 页 ， 用 户 的 动作 总 是 与 服务 器 进行 同步 
思考 ， 客 户 在 网 页 上 的 操作 转化 为 Http 请 求 传 回 服务 器 ， 而 服务 器 接受 请 求 以 及 相关 数据 ， 然 后 解析 数 
据 并 将 其 发 送 给 相应 的 处 理 单元 后 ， 将 返回 的 数据 转 成 HTML 页 返还 给 客户 。 而 当 服 务 器 处 理 数据 时 ， 
用 户 只 能 等 待 ， 每 一 步 操 作 都 需要 等 待 服务 器 返回 新 的 网 页 。 由 于 每 次 应 用 的 交互 都 需要 向 服务 器 发 送 
请 求 ， 应 用 的 响应 时 间 就 依赖 于 服务 器 的 响应 时 间 ， 这 就 导致 了 用 户 页 面 的 响应 比 本 地 应 用 慢 得 多 。 

运用 了 Ajax 技术 的 Web 应 用 模型 , 它 的 工作 原理 相当 于 在 客户 端 和 服务 器 端 之 间 添 加 了 一 个 中 间 
层 ， 称 为 Ajax 引擎 (采用 JavaScript 编写 ， 通 常 在 一 个 隐藏 的 框架 中 )， 实 现 了 与 服务 器 进行 异步 思考 
的 通信 能 力 ， 从 而 使 用 户 从 请 求 /响应 的 循环 中 解脱 出 来 ， 向 服务 器 发 出 异步 请 求 ， 也 就 是 不 用 等 待 服 
务 器 的 通信 。 所 以 用 户 不 用 再 打开 一 个 空白 窗口 ， 等 待 服务 器 完成 后 再 进行 响应 。Ajax 应 用 可 以 仅 向 
服务 器 发 送 并 取 回 必需 的 数据 ， 它 使 用 SOAP 或 其 他 一 些 基 于 XML 的 Web Service 接口 ， 并 在 客户 端 
采用 JavaScript 处 理 来 自 服务 器 的 响应 。 因 为 在 服务 器 和 浏览 器 之 间 交 换 的 数据 大 量 减 少 ， 所 以 Web 
站 点 看 起 来 是 即时 响应 的 。 同 时 很 多 的 处 理工 作 可 以 在 发 出 请 求 的 客户 端 机 器 上 完成 ， 所 以 Web 服务 
器 的 处 理 时 间 也 减少 了 。 

引入 Ajax 的 Web 模型 与 传统 的 Web 模型 比较 如 图 4.30 所 示 。 


《传统 Web 应 用 模型 > 


Ajax Web 应 用 模型 > 


图 4.30 传统 Web 应 用 模型 与 Ajax Web 应 用 模型 的 比较 
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4.11.4 ”Ajax 的 工作 流程 


使 用 Ajax， 用户 可 以 创建 接近 本 地 桌面 应 用 的 更 直接 、 更 可 用 、 更 丰富 、 更 动态 的 Web 用 户 界面 。 
Ajax 内 部 的 工作 流程 如 图 4.31 所 示 。 


加 a | 多 eb 浏览 器 发 送 HTTP 请 求 给 服务 器 展 


一 一 、 4 
多 eb 服务 器 格 请 求 使 用 HTML 和 JavaScript 响 应 
Intemet 用 户 Web 客户 端 Web 服 务 器 


JavaScript 代 码 向 服务 器 发 送 不 可 见 调用 请 求 1 
| 


服务 器 通过 发 送 
请 求 的 数据 进行 响应 

:更 新 客户 端 显 示 的 
内 容 


| 
| 
| 
| 
| 
| 
| 


JavaScript 根 据 响应 数据 更 新 Web 页 面 


Ge 


4.31 Aijax 内 部 工作 流程 图 
4.11.5_ Ajax 中 的 核心 技术 XMLHttpRequest 


Ajax 技术 中 最 核心 的 技术 就 是 XMLHttpRequest， 它 是 一 个 具有 应 用 程序 接口 的 JavaScript 对 象 ， 
能 够 使 用 超 文本 传输 协议 CHTTP ) 连接 一 个 服务 器 ， 是 微软 公司 为 了 满足 开发 者 的 需要 ， 于 1999 年 
在 正 5.0 浏 览 器 中 率先 推出 的 。 现 在 许多 浏览 器 都 对 其 提供 了 支持 ， 不 过 实现 方式 与 IE 有 所 不 同 。 

通过 XMLHttpRequest 对 象 ，Ajax 可 以 像 桌面 应 用 程序 一 样 只 同 服务 器 进行 数据 层面 的 交换 ， 而 
不 用 每 次 都 刷新 页 面 ， 也 不 用 每 次 都 将 数据 处 理 的 工作 交 给 服务 器 来 做 ， 这 样 既 减轻 了 服务 器 负担 ， 
又 加 快 了 响应 速度 ， 而 且 还 缩短 了 用 户 等 待 的 时 间 。 

在 使 用 XMLHttpRequest 对 象 发 送 请 求 和 处 理 响 应 之 前 ， 首 先 需要 初始 化 该 对 象 ， 由 于 
XMLHttpRequest 不 是 一 个 W3C 标准 ， 所 以 对 于 不 同 的 浏览 器 ， 初 始 化 的 方法 也 是 不 同 的 。 

1. IE 浏览 器 

IE 浏览 器 把 XMLHttpRequest 实例 化 为 一 个 ActiveX 对 象 。 具 体 方法 如 下 : 

var http_request = new ActiveXObject("Msxml2.XMLHTTP"); 
或 者 


var http_request = new ActiveXObject("Microsoft.XMLHTTP"); 


py 
于 人 昌 上 面 语法 中 的 Msxml2.XMLHTTP 和 Microsoft.XMLHTTP 是 针对 IE 浏览 器 的 不 同 版 本 而 
进行 设置 的 ， 目 前 比较 常用 的 是 这 两 种 。 
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2. Mozilla、Safari 等 其 他 浏览 器 
Mozilla、Safari 等 其 他 浏览 器 把 它 实例 化 为 一 个 本 地 JavaScript 对 象 。 具 体 方法 如 下 : 
var http_request = new XMLHttpRequest(); 


为 了 提高 程序 的 兼容 性 ， 可 以 创建 一 个 跨 浏 览 器 的 XMLHttpRequest 对 象 。 创建 一 个 跨 浏 览 器 的 
XMLHttpRequest 对 象 其 实 很 简单 ， 只 需要 判断 一 下 不 同 浏 览 器 的 实现 方式 ， 如 果 浏 览 器 提供 了 
XMLHttpRequest 类 ， 则 直接 创建 一 个 实例 ， 和 否则 使 用 正 的 ActiveX 控件 。 有 具体 代码 如 下 : 


if (window.XMLHttpRequest) {// Mozilla、Safari… 
http_request = new XMLHttpRequest(); 
} else if (window.ActiveXObject) { /1E 浏览 器 
try{ 
http_request = new ActiveXObject("Msxml2.XMLHTTP"); 
} catch (e){ 
ty{ 
http_request = new ActiveXObject("Microsoft.XMLHTTP"); 
} catch (e) {0} 
} 
} 


4 
说 明 由 于 JavaScript 具有 动态 类 型 特性 ， 而 且 XMLHttpRequest 对 象 在 不 同 浏览 器 上 的 实例 是 
兼容 的 ， 所 以 用 同样 的 方式 访问 XMLHttpRequest 实例 的 方法 即 可 ， 不 需要 再 考虑 创建 该 实例 的 方 
法 是 什么 。 


至 此 ， 一 个 完整 的 网 络 在 线 考试 系统 已 经 全 部 完成 。 在 程序 的 开发 过 程 中 ， 采 用 了 switch 框架 ， 
使 整个 系统 的 设计 思路 更 加 清晰 。 同 时 ， 为 了 使 程序 更 加 人 性 化 ， 系 统 中 应 用 了 Ajax 技术 实现 在 线 考 
试 时 自动 计时 等 功能 。 和 希望 读者 能 认真 学 习 ， 并 做 到 融会 贯通 。 


全 各 立 - 


下 里 
物流 配送 信息 网 


(Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


物流 信息 化 是 指 物流 企业 运用 现代 信息 技术 对 物流 过 程 中 产生 的 
全 部 或 部 分 信息 进行 和 采集、 分类、 传递 、 汇 总 、 坦 询 等 一 系列 处 理 活 
动 ， 以 实现 对 货物 流动 过 程 的 控制 ， 从 而 降低 成 本 、 提 高 效益 的 管理 
活动 。 

物流 企业 信息 化 的 目的 就 是 要 满足 企业 自身 管理 的 需要 和 不 同类 
型 企业 在 物流 业务 外 包 过 程 中 对 信息 交换 方 的 要 求 ， 也 就 是 通过 建设 
物流 信息 系统 ， 提 高 信息 流转 效率 ， 降 低 物流 运作 成 本 。 

目前 我 国正 处 于 全 面 推进 信息 化 的 进程 之 中 ， 所 以 物流 领域 的 信 
息 化 已 成 为 一 个 必然 。 物 流 信 息 化 将 成 为 现代 物流 的 灵魂 ， 将 是 现代 
物流 发 展 的 必 经 之 路 。 

本 章 开发 的 物流 配送 信息 网 就 是 为 了 增强 企业 的 市 场 竞争 力 ， 加 
快 企业 的 信息 化 进程 ， 使 企业 在 市 场 竞争 中 立 于 不 败 之 地 。 通 过 阅读 
本 章 ， 可 以 学 习 到 : 
物流 配送 系统 开发 的 基本 过 程 
Nm 如 何 创建 MySQL 数据 库 的 存储 过 程 
如 何 通 过 MySQL 数据 库存 储 过 程 实现 用 户 登 录 
如 何 实现 数据 库 中 数据 的 模糊 查询 技术 
如 何 创 建 浮动 框架 
如 何 实现 订单 的 打印 技术 


于 


各 吾 吾 至 
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51, 浊 发 背景 


随 着 我 国信 息 化 进程 的 全 面 推进 ， 各 领域 的 信息 化 进程 都 在 飞速 的 发 展 ， 同 样 也 推动 着 物流 领域 
的 信息 化 进程 飞快 的 向 前 发 展 。 由 于 信息 化 进程 的 全 面 推进 ， 对 现代 物流 提出 更 高 的 要 求 :信息 化 、 
自动 化 、 网 络 化 、 智 能 化 和 柔性 化 等 。 物 流行 业 的 竞争 日 益 激烈 ， 客 户 需 求 的 标准 也 越 来 越 高 ， 物 流 
企业 要 想 在 市 场 中 占有 一 席 之 地 ， 必 须要 建立 一 个 高 效 、 快 捷 、 方 便 的 物流 信息 系统 ， 为 客户 提供 一 
流 的 服务 ， 并 且 能 够 及 时 、 准 确 地 掌握 客户 的 需求 ， 对 客户 的 需求 做 出 快速 的 反映 ， 在 最 短 的 时 间 内 
以 最 大 限度 挖掘 和 优化 物流 资源 来 满足 客户 的 需求 ， 从 而 建立 高 效 的 数字 化 物流 经 济 。 


5.2 系统 分 析 


5.2.1 需求 分 析 


随 着 现代 物流 信息 化 进程 的 加 快 ， 传 统 的 物流 管理 方式 已 经 不 再 适应 当前 物流 发 展 的 要 求 ， 取 而 
代 之 的 将 是 以 计算 机 为 基础 的 网 络 化 物流 管理 方式 。 某 物流 配送 公司 为 适应 物流 信息 化 进程 的 发 展 ， 
急需 开发 一 个 物流 配送 系统 ， 通 过 网 络 来 实现 对 物流 操作 流程 进行 管理 ， 不 但 可 以 为 企业 的 运营 过 程 
节省 大 量 的 人 力 、 物 力 、 财 力 和 时 间 ， 提 高 物流 系统 运行 的 效率 ， 而 且 可 以 使 企业 在 客户 心中 树立 了 
一 个 全 新 的 形象 ， 为 企业 的 发 展 英 定 一 个 良好 的 基础 。 现 根据 对 该 物流 配送 公司 的 实际 调查 ， 以 及 公 
司 的 具体 要 求 ， 制 定 出 物流 配送 信息 网 的 规划 方案 。 具 体内 容 如 下 


回 ”网 站 页 面 设计 要 求 美观 大 方 ， 能 够 展示 企业 形象 。 
回 ”为 企业 在 客户 中 树立 一 个 全 新 的 形象 。 

回 ”网 站 的 操作 流程 简单 、 方 便 ， 能 够 提高 工作 效率 。 
回 ”提供 物流 配送 的 全 程 跟踪 。 

回 ”提供 配送 信息 的 及 时 查询 。 

回 ”实现 对 配送 车 辆 的 管理 。 

回 ”实现 对 客户 信息 的 管理 。 

回 ”实现 发 货 单打 印 的 功能 。 


5.2.2 可 行 性 分 析 

物流 配送 信息 网 的 开发 不 但 能 使 物流 企业 走 上 科学 化 、 网 络 化 管理 的 道路 ， 而 且 能 够 为 企业 带 来 
巨大 的 经 济 效益 和 技术 上 飞速 的 发 展 。 

1. 经 济 性 

科学 的 管理 方法 ， 便 捷 的 操作 环境 ， 系 统 的 经 营 模式 ， 将 为 企业 带 来 更 多 的 客户 资源 ， 树 立 企业 
的 品牌 形象 ， 提 高 企业 的 经 济 效益 。 

2. 技术 性 

网 络 化 的 物流 管理 方式 ， 在 操作 过 程 中 能 够 快捷 地 查找 出 车 源 信息 、 客 户 订单 以 及 客户 信息 ; 能 
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够 对 货物 进行 全 程 跟踪 ， 了 解 货物 的 托运 情况 ， 从 而 使 企业 能 够 根据 实际 情况 ， 做 好 运营 过 程 中 的 各 
项 准备 工作 ， 作 出 及 时 准确 的 调整 ， 能够 保证 托运 人 以 及 收 货 人 对 货物 进行 及 时 的 处 理 。 


5.3 系统 设计 


5.3.1 系统 目标 


结合 目前 网 络 上 物流 配送 系统 的 设计 方案 ， 对 客户 做 的 调查 结果 以 及 企业 的 实际 需求 ， 本 项 目 在 
设计 时 应 该 满足 以 下 目标 : 
界面 设计 美观 大 方 、 操 作 简单 。 
功能 完善 、 结 构 清晰 。 
能 够 快速 查询 车 源 信息 。 
能 够 准确 填写 订单 。 
能 够 实现 订单 查询 、 打 印 。 
能 够 实现 对 回 单 处 理 。 
能 够 对 车 源 信息 进行 添加 、 修 改 和 删除 。 
能 够 对 客户 信息 进行 管理 。 
能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 
良好 的 数据 库 系 统 支持 。 
系统 运行 稳定 ， 具 备 良好 的 防范 措施 。 


系统 功能 结构 
结合 需求 分 析 和 系统 目标 中 的 内 容 ， 物 流 配送 信息 网 的 功能 结构 已 经 设计 完成 。 为 了 使 读者 能 够 


更 清楚 地 了 解 网 站 的 结构 ， 下 面 给 出 物流 配送 信息 网 的 功能 模块 结构 图 和 工作 流程 图 。 
物流 配送 信息 网 的 功能 模块 结构 图 如 图 5.1 所 示 。 


因 因 办 办 办 办 办 办 办 办 轨 


a 
〇 
i 


一 | 添加 客户 
| 删除 客户 
| 添加 车 源 信息 


查询 车 源 信息 上 -1 
可 | 修改 车 源 信息 
使 用 该 车 “| 
删除 车 源 信息 


图 5.1 物流 配送 信息 网 功能 模块 结构 图 
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物流 配送 信息 网 的 工作 流程 图 如 图 5.2 所 示 。 


i 客户 提交 订单 
客户 
登录 
管理 员 


图 5.2 物流 配送 信息 网 的 工作 流程 图 


5.3.3 ”系统 预览 


物流 配送 信息 网 由 多 个 程序 页 面 组 成 ， 下 面 给 出 儿 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 
物流 配送 信息 网 主页 如 图 5.3 所 示 ， 该 页 面 用 于 展示 本 系统 的 车 源 信息 查询 模块 。 登 录 页 面 如 图 5.4 
所 示 。 


Es PEI SONG XIN XI 
mad, Em2L SenARN 
| 
Ce 
物流 配送 信息 网 

轩 用 记名 rr 定 码 , 665 医 基 有 iss) 
| anueen am 

图 5.3 主页 (光盘 \TM\05\wlgl\indexs.php) 图 5.4 登录 页 面 (光盘 \TM\05\wlglindex.php) 


回执 发 货 单 确认 模块 的 页 面 效 果 如 图 5.5 所 示 , 该 页 面 主要 用 于 实现 对 回执 发 货 单 的 处 理 。 发 货 单 
查询 模块 的 页 面 效 果 如 图 5.6 所 示 , 该 页 面 主要 用 于 实现 对 指定 发 货 单 的 查询 ， 并且 输 出 发 货 单 的 详细 
内 容 。 

车 源 信息 管理 模块 的 页 面 效果 如 图 5.7 所 示 ， 该 页 面 实现 对 车 源 信息 的 添加 、 修 改 和 删除 操作 。 修 
改 管理 员 密 码 模块 的 页 面 效果 如 图 5.8 所 示 ， 该 页 面 实现 对 管理 员 密码 的 修改 。 


@ 
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5.7 车 源 信息 管 理 (光盘 \TM\05\wlgl\car.php) ”图 5.8 修改 管理 员 密码 〈 光 盘 \TMMWO5\ wlgl\update_pass.php) 
5.3.4 开发 环境 


在 开发 物流 配送 信息 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows Server 2003/Linux (推荐 )。 
服务 器 : Apache 2.2.8。 

PHP 软件 : PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.10.3。 
开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辨 率 : 最 佳 效 果 为 1024x768 像素 。 


客户 端 
浏览 器 : IE 6.0 及 以 上 版 本 。 


办 办 办 办 办 办 办 轨 


罗 -人 


对 ”PHP 项 目 开发 全 程 实录 (第 3 版 ) 
回 ”分辩 率 : 最 佳 效果 为 1024x768 像素 。 
5.3.5 文件 夹 组 织 结构 
在 进行 网 站 开发 之 前 ， 要 对 网 站 的 整体 文件 夹 组 织 架 


构 进行 规划 。 对 网 站 中 使 用 的 文件 进行 合理 的 分 类 ， 分 别 
放置 于 不 同 的 文件 夹 下 。 通 过 对 文件 夹 组 织 架构 的 规划 ， 


= 
可 以 确保 网 站 文件 目录 明确 、 条 理 清晰 ， 同 样 也 便于 网 站 Pn 
后 期 的 更 新 和 维护 。 物 流 配送 信息 网 的 文件 夹 组 织 结构 如 
图 5.9 所 示 。 图 5.9 文件 夹 组 织 结构 


5.4 ”数据 库 设计 


物流 配送 信息 网 必须 拥有 数据 库 的 支持 ， 所 有 物流 配送 的 数据 都 应 该 存储 到 数据 库 中 ， 便 于 管理 
员 查 找 车 辆 、 订 单 和 客户 的 信息 。 如 果 没 有 数据 库 的 支持 ， 那 么 物流 配送 信息 网 将 没有 任何 意义 。 本 
节 将 对 物流 配送 信息 网 的 数据 库 设计 进行 详细 介绍 。 
5.4.1 ”数据 库 分 析 


物流 配送 信息 网 是 一 个 中 小 型 的 企业 管理 系统 ， 考 虑 到 开发 的 成 本 、 搭 配 的 合理 性 以 及 操作 的 灵 
活性 等 ， 使 用 MySQL 数据 库 是 最 佳 的 选择 。MySQL 数据 库 是 完全 免费 的 ， 使 用 它 不 需要 任何 费用 ， 
可 以 直接 从 网 上 免费 下 载 ; MySQL 数据 库 的 操作 也 非常 方便 ， 不 但 可 以 在 命令 模式 下 操作 ， 而 且 配备 
目前 比较 流行 的 图 形 化 管理 工具 phpMyAdmin， 能 够 轻松 地 实现 对 MySQL 数据 库 的 管理 和 操作 。 


5.4.2 ”数据 库 概 念 设计 

根据 上 述 各 节 对 物流 配送 信息 网 做 的 需求 分 析 和 系统 设计 ,整理 出 物流 配送 信息 网 的 实体 关系 E-R 
图 。 其 中 包括 管理 员 信息 实体 、 车 源 信息 实体 、 车 辆 日 志 信息 实体 、 客 户 信息 实体 和 发 货 单 信息 实体 。 

1. 管理 员 信息 实体 

管理 员 信 息 实体 用 于 存储 管理 员 的 登录 名 称 和 密码 信息 ， 包 括 用 户 名 和 密码 两 项 内 容 。 管 理 员 信 
息 实体 的 E-R 图 如 图 5.10 所 示 。 


用 
图 5.10 管理 员 信 息 实体 的 E-R 图 
2. 车 源 信息 实体 
车 源 信息 实体 用 于 存储 企业 拥有 的 车 辆 信息 ， 包 括 车 主 姓 名 、 车 主 身份 证 号 、 车 牌号 码 、 车 主 联 
系 电话 、 车 主 家 庭 地 址 、 车 辆 行驶 路 线 和 车 辆 描述 。 车 源 信息 实体 的 E-R 图 如 图 5.11 所 示 。 


第 5 章 物流 配送 信息 网 (ApachetPHP+phpMyAdmin+MySQL 5.0 实现 ) 


车 主 家 庭 地址 


车 牌 


号 码 


车 辆 行驶 路 线 


车 主 联系 电话 


图 5.11 车 源 信息 实体 的 E-R 图 


3. 车 辆 日 志 信 息 实体 

车 辆 日 志 信息 实体 记录 车 辆 当前 是 否 被 占用 ， 以 及 使 用 的 时 间 、 执 行 的 任务 ， 包 括 车 牌号 码 、 车 
辆 日 志 信息 (详细 的 车 辆 使 用 描述 )、 日 志 时 间 和 发 货 单 ID ( 即 执行 的 任务 )。 车辆 日 志 信息 实体 的 E-R 
图 如 图 5.12 所 示 。 


4. 客户 信息 实体 
客户 信息 实体 用 于 存储 客户 的 信息 ， 包 括 客户 姓名 、 客 户 电话 和 客户 联系 地 址 。 客 户 信息 实体 的 


E-R 图 如 图 5.13 所 示 。 


军 辆 日 志 信息 (人 请 ) | 客户 信息 | 客户 联系 地 址 
图 5.12 车 辆 日 志 信息 实体 的 E-R 图 图 5.13 客户 信息 实体 的 E-R 图 
5. 发 货 单 信息 实体 


发 货 单 信息 实体 用 于 存储 客户 填写 的 发 货 单 信息 ， 包 括 车 牌号 码 、 车 主 电话 、 货 物 描述 、 发 货 人 、 
发 货 时 间 、 发 货 人 电话 、 发 货 地 址 、 收 货 人 、 收 货 人 电话 、 收 货 地 址 和 付款 方式 。 发 货 单 信息 实体 的 


E-R 图 如 图 5.14 所 示 。 
CO Cr > 
人 地址 


CC 年 二 电话 发 货 单 信息 收费 地 址 > 
图 5.14 发 货 单 信息 实体 的 E-R 图 
5.4.3 ”创建 数据 库 及 数据 表 


在 物流 配送 信息 网 中 应 用 的 是 db_wlgl 数据 库 ， 其 中 涉及 5 个 数据 表 ， 如 图 5.15 所 示 。 
下 面 介绍 物流 配送 信息 网 中 使 用 的 数据 表 。 

1. tb_admin (管理 员 信息 表 ) 

管理 员 信息 表 主 要 用 于 存储 管理 员 的 登录 名 和 密码 。 该 数据 表 的 结构 如 图 5.16 所 示 。 
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胃 服务 器 : localhost ; 部 数据 库 : db_wigl 

表 类 型 要 昌 说 明 
th_admin MISAM 
MYISAM 
WISAM 
WISAM 
MISAM 


gb2312_chinese_ ci 
gb2312_chinese_di 
gb2312_chinese_di 
gb2312_chinese_di 
gb2312_chinese_ci 发 半音 信息 表 


图 5.15 物流 配送 信息 网 的 数据 库 
2. tb_car (车 源 信 息 表 ) 


tb_car 
th_car log 
‘th_customer 
th_shopping 


胃 服务 器 : localhost 总 数据 库 : db_wlgl 国 表 :tb_admin 


字段 类 型 整理 尾 性 Null 默认 格外 说 明 
到 ini0) 否 auto_increment 管理 员 信息 表 的 主妇 
admin_user varchat50) gb2312_chinese_ci 要 区 录 名 称 
admin_pass varchat50) gb2312_chinese_cl 村 蕉 录 密码 


5.16 管理 员 信息 表 结构 


车 源 信息 表 主 要 用 于 存储 配送 公司 拥有 的 车 辆 信息 ， 主 要 包括 车 主 姓名 、 车 主 的 身份 证 号 码 、 车 
牌号 码 、 车 主 电话 、 车 主 联 系 地 址 、 车 辆 行驶 路 线 和 车 辆 描述 。 该 数据 表 的 结构 如 图 5.17 所 示 。 
3. tb_car_log (车 辆 日 志 信息 表 ) 


车 辆 日 志 信息 表 主 要 用 于 存储 车 辆 的 使 用 情况 信息 ， 主 要 包括 车 牌号 码 、 车 辆 日 志 信息 、 日 志 创 
建 时 间 和 发 货 单 的 ID。 该 数据 表 的 结构 如 图 5.18 所 示 。 


轩 服务 器 ;localhost ， 品 数据 库 : db_wlgl ;加 表 :tb_ear 
3 天 要 理 属性 


a intto) 要 
Username varcher(s0) gb2312_ohineso_l 理 
ser_numher varchar(50) 理 
car_number varchar(5o) 理 
‘tel varchar(s0) 否 
adress Vanartao) 百 
car_road mediumiex 理 
car_contont 。 mediumlat gb2312_chinese_el 再 


图 5.17 车 源 信息 表 结构 
4. tb_customer (客户 信息 表 ) 


auto_increment 车 晨 过 息 示 的 主键 


车 主 的 蛤 各 


夺 寺 折射 向 证 3 码 国 要 各 法 localhost， 司 数 包 库 ; db_wigi， 画 表 :由 -car log 

ES Fn em 碍 Nl 守 和 外 加 
Cri 百 un_neament 天 Br 

ss (car manbor wa W232 towioH 。。 理 a 

ENR coriog moourter gp2212 chinese 9 再 天 日 本 

所 oa sa。 cam 机 Eri 

二 anen jg。 nars0 mpzal2_cnnese a 再 esp 


图 5.18 车辆 日 志 信息 表 结构 


客户 信息 表 主 要 用 于 存储 客户 的 信息 ， 包 括 客户 姓名 、 客 户 电话 以 及 联系 地 址 。 该 数据 表 的 结构 


如 图 5.19 所 示 。 


5. tb_shopping (发 货 单 信息 表 ) 


发 货 单 信息 表 主要 用 于 存储 客户 填写 的 发 货 单 中 的 信息 ， 主 要 包括 车 牌号 码 、 车 主 电话 、 货 物 描 
述 、 发 货 人 、 发 货 时 间 、 发 货 人 电话 、 发 货 地 址 、 收 货 人 、 收 货 人 电话 、 收 货 地 址 和 付款 方式 。 该 数 


据 表 的 结构 如 图 5.20 所 示 。 


胃 服务 器 : losalhest > 局 数据 库 : db_wigl ， 到 圾 : tb_shopping 
人 A 整理 属性 Nal 歌 认 全 外 说 明 


中 mn 四 再 MU 
NN [ELSEEH EEE 画 ER 
ahun_cantent mmediurntad gb2312_chinese_ci 百 其 析 指 壕 
ao Hd varchar(50) 902312_chinsss_ci 可 发 人 7 
ahuo_user Varchart50) ge2312_chineoe_ el 至 类 从 入 
Tahwo tmo me 再 发 让 呵 间 
ee em 
加 有 和 本 -ocalhost》 避 要 所 库 :db_wigl 》 加 家 tb_customer PR 
于 芜 要 | Nel MA 本 je shouhuo_user Gychar50) go2312_chinese_cl 下 收 符 人 
和 可 I showuo adgross medumtat qo2312 Chinsca_cl 再 岗 持 地 址 
5 再 ee | | 
Customer tel vartharG9) gb3312_ehinese_ti 百 至 P 电 二 rm [一 一人 pr 百 pr 
ustomer_address varchar(30) 9b2312_chinese_ci 否 联系 地 让 shouhun_te! machant50) 2312_chiness_ci 至 中 全 和 电话 


5.19 客户 信息 表 结 构 


@ 


图 5.20 发 货 单 信息 表 结 构 
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5.5 ”网 站 首页 设计 


5.5.1 网 站 首页 概述 


网 站 首页 是 整个 网 站 的 门面 ， 是 浏览 者 看 到 的 第 一 视觉 界面 ， 所 以 在 设计 网 站 的 首页 时 应 该 将 网 

站 中 主要 的 内 容 尽量 展示 给 浏览 者 ， 让 浏览 者 能 够 更 快 地 了 解 网 站 的 内 容 。 物 流 配 送信 息 网 的 首页 主 
要 包括 如 下 功能 。 
车 源 信息 查询 : 主要 实现 车 辆 信息 查询 ， 及 时 为 客户 选择 货物 配送 的 方案 。 
发 货 单 : 主要 用 于 客户 发 货 单 信息 填写 。 
回执 发 货 单 确认 : 主要 用 于 对 货物 配送 回执 单 的 确认 。 

发 货 单 查询 : 主要 用 于 对 发 货 单 进行 查询 。 

客户 信息 管理 : 主要 用 于 对 客户 的 信息 进行 管理 。 

车 源 信息 管理 : 主要 用 于 对 企业 拥有 的 车 辆 信息 进行 管理 。 

修改 密码 : 主要 用 于 对 管理 员 登 录 的 密码 进行 修改 。 
案例 中 提供 的 网 站 首页 如 图 5.21 所 示 。 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\05\wlgl\index.php。 


站 因 因 因 因 办 办 办 
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物 南 呈送 信息 网 ， 舍 蕉 之 上 ， 勇 力 打造 物 砚 品 中 


w 网 站 Flash 头 文件 
SE | L 


功能 模块 内 容 展 示 


图 5.21 物流 配送 信息 网 首页 
5.5.2 ”网 站 首页 技术 分 析 
物流 配送 信息 网 是 为 物流 配送 公司 设计 的 一 个 物流 管理 系统 ， 其 页 面 的 设计 突出 简洁 、 方 便 ， 功 


能 的 实现 以 便于 操作 和 维护 为 根本 。 网 站 首页 的 页 面 设计 如 图 5.21 所 示 ， 由 一 个 简单 的 框架 组 成 ， 其 
结构 由 两 部 分 组 成 : 一 部 分 用 于 输出 网 站 中 包括 的 模块 ， 另 一 部 分 用 于 输出 对 应 模块 中 的 内 容 。 首 页 


框架 结构 如 图 5.22 所 示 。 


PHP 项 目 开发 全 程 实录 (第 3 版 ) 


输出 功能 模块 的 内 容 


5.22 首页 框架 结构 


这 里 使 用 的 框架 结构 不 是 frame 框架 ， 而 是 应 用 switch 语句 ， 加 上 表格 的 嵌 套 实现 的 框架 效果 。 
有 关 switch 语句 的 详细 应 用 可 以 参考 2.12.2 节 


5.5.3 ”网 站 首页 的 实现 过 程 


物流 配送 信息 网 的 首页 设计 非常 简单 ， 主 要 应 用 一 个 简单 的 框架 功能 ， 并 且 以 超 链 接 作为 辅助 ， 
最 终 实现 在 不 同 功能 模块 之 间 的 跳 转 。 

倒 性 01 ”代码 位 置 ， 光盘 \TM\05\wlgl\indexs.php 

首先 连接 数据 库 ， 判 断 登 录用 户 的 身份 ， 如 果 是 管理 员 ， 则 可 以 进入 网 站 进行 操作 ;否则 将 提示 

请 您 正确 登录 !”， 并 且 跳 转 到 网 站 的 登录 页 面 。 关 键 代码 如 下 : 


<?php session_start(); /初始化 session 变量 

include("conn/conn.php"); /| 连接 数据 库 

if($_SESSION[admin_user]==true){ 1/ 判断 登录 用 户 的 身份 

?> 

<!-- 省 略 了 部 分 代码 --> 

<?php 

}else{ // 如 果 没有 以 管理 员 登 录 ， 则 跳 转 到 登录 页 
echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>"; 

i 
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然后 设计 首页 中 使 用 的 模块 超 链接 ， 并 且 为 每 个 超 链接 定义 变量 标识 ， 当 单 击 不 同 功能 模块 超 链 
接 时 提交 不 同 的 变量 标识 ， 作 为 下 一 步 获取 模块 中 详细 内 容 的 依据 ， 其 中 还 使 用 ononMouseOver 和 
onMouseOut 事件 ， 控 制 鼠 标 移动 到 和 离开 单元 格 时 显示 不 同 的 颜色 。 部 分 代码 如 下 : 


<!- -一 -一 一 一 一 -一 一 一 应 用 onMouseOver 和 onMouseOut 事 件 控制 鼠标 移 过 和 离开 时 的 时 间 一 一 -一 一 -一 一 - 一 > 
<tr onMouseOver="this.bgColor='#9FCB3A"onMouseOut="this.bgColor=#F5F5F5" > 
<td height="28" align="center">&nbsp;</td> 
I 设置 车 源 信息 查 询 模块 的 超 链接 ， 变 量 标识 为 "Imbs= 车 源 信息 查询 "一 -一 一 -一 -> 
<td align="left"><a href="indexs.php?Imbs= 车 源 信 息 查 询 "> 车 源 信息 查询 </a></td> 
</tr> 
<tr onMouseOver="this.bgColor='#9FCB3A"onMouseOut="this.bgColor=#F5F5F5" > 
<td height="28" align="center">&nbsp;</td> 
< 上-- -一 一 一 一 一 一 一 一 一 一 一 设置 发 货 单 模块 的 超 链接 ， 变 量 标识 为 "Imbs= 发 货 单 "一 一 一 一 一 一 一 一 一 一 一 - -> 
<td align="left"><a href="indexs.php?Imbs= 发 货 单 "> 发 货 单 </a></td> 
</tr> 


最 后 应 用 switch 语句 ， 根 据 变量 标识 Slmbs 提交 的 值 进行 判断 ， 使 用 include 包含 语句 调用 不 同 功 


GS 
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模块 的 脚本 文件 。 关 键 代码 如 下 : 


<?php 
switch($ImbsX{ // 获 取 变 量 $Imbs 的 值 
case "车 源 信息 查询 ": /判断 当 变 量 $Imbs 的 值 是 "车 源 信息 查询 "时 输出 下 面 的 内 容 
include("car_select.php"); /1 调用 指定 的 包含 文件 
break; /跳出 循环 
// 这 里 省 略 了 部 分 代码 
case ™: /判断 当 变 量 的 值 为 空 时 
include("car_select.php"); // 调 用 该 文件 
break; /跳出 循环 
} 
?> 


5.6 ”车 源 信 息 查 询 模 块 设计 


5.6.1 车 源 信息 查询 模块 概述 


车 源 信息 查询 模块 的 功能 是 根据 客户 提供 的 货 
物 配送 的 地 点 , 从 数据 库 中 查询 有 关 该 路 线 的 车 辆 ， 
为 客户 提供 一 个 合理 配送 的 路 线 ， 最 后 确定 发 货 订 
单 。 操 作 流 程 如 图 5.23 所 示 。 


5.6.2 ”车 源 信息 查询 模块 技术 分 析 


车 源 信息 查询 模块 的 主要 功能 就 是 查询 车 辆 的 
使 用 信息 ， 为 客户 提供 最 合适 的 货物 配送 路 线 。 其 
中 应 用 的 关键 技术 自然 就 是 查询 方法 ， 为 了 给 客户 
提供 最 合适 、 最 满意 、 最 快捷 的 配送 服务 ， 这 里 使 
用 的 是 模糊 查询 技术 。 

通过 模糊 查询 技术 ， 只 要 客户 提出 配送 货物 的 出 发 地 点 和 到 达 地 点 ， 管 理 员 就 可 以 从 数据 库 中 提 
取出 所 有 与 该 路 线 相关 的 车 辆 信息 ， 包 括 车 辆 的 承载 能 力 、 车 辆 是 否 被 占用 和 车 辆 的 使 用 情况 ， 客 户 
可 以 根据 实际 的 情况 〈 时 间 、 货 物 数 量 、 路 线 等 ) 选择 车 辆 ， 确 认 后 填写 发 货 单 。 

模糊 查询 技术 的 实现 是 在 car_select.php 文件 中 完成 的 ， 首 先 要 与 数据 库 建 立 连接 ; 然后 创建 一 个 
form 表单 ,设置 两 个 文本 框 selectl 和 select2 用 于 提交 出 发 地 点 和 到 达 地 点 ,将 表单 的 值 提交 到 当前 页 ; 
最 后 编写 PHP 脚本 语句 ， 以 表单 中 获取 的 数据 为 条 件 ， 执 行 like 模糊 查询 ， 从 数据 库 的 指定 表 中 查询 
符合 条 件 的 信息 ， 并 且 将 查询 出 的 信息 显示 到 页 面 中 。 程 序 的 关键 代码 如 下 : 

倒 程 02 代码 位 置 ， 光盘 \TM\05\wlgl\car_select.php 


图 5.23 车 源 信息 查询 的 操作 流程 


<?php 
session_start(); include("conn/conn.php"); /连接 数据 库 
?> 
<!- 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 -创建 form 表单 ， 提 交 出 发 地 点 和 到 达 地 点 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 -- > 
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<form id="form1" name="form1" method="post" action="indexs.php?Imbs= 车 源 信息 查询 " onSubmit="javascript: 
return check_select_car();"> 
<td height="30" colspan="4" align="right" bgcolor="#FFFFFF"> 查 询 
© <input name="select1" type="text" id="select1" size="10"> 至 
@ <input name="select2" type="text" id="select2" size="10"></td> 
<td colspan="2" bgcolor="#FFFFFF"><input type="submit" name="Submit" value=" 提 交 " /></td> 
</form> 
< = 获取 表单 提交 的 值 ， 根 据 该 条 件 执行 like 模糊 查询 ， 从 数据 库 中 读 取 数 据 -一 一 -- > 
<?php 
© if($select1==true || $select2==true}{ // 判 断 表单 提交 的 值 是 否 为 真 
产 te 单 提交 的 值 为 条 件 ， 从 数据 表 中 查询 数据 "ee wh 
®@ $query="select * from tb_car where car_road like "%$select1%' and car_road like '%$select2%"; 


$result=mysql_query($query); /| 执行 查询 语句 
if(mysql_num_rows($result)>0X{ /| 判断 是 否 存 在 符合 条 件 的 数据 
while($myrow=mysql_fetch_array($result)X ”// 应 用 while 循环 语句 ， 输 出 符合 条 件 的 数据 
?> 
<tr> 
<td height="26" align="center" bgcolor="#FFFFFF"><?php echo $myrow[car_number];?></td> 
<td bgcolor="#FFFFFF"><?php echo $myrow[car_road];?></td> 
<td colspan="2" bgcolor="#FFFFFF"><?php echo $myrow[car_content];?></td> 
<td bgcolor="#FFFFFF"> 
<?php 
/* eexxx* 根 据 车 牌号 码 ， 从 车 辆 日 志 信息 表 中 读 取 对 应 车 辆 的 使 用 信息 ********* */ 
$querys=mysql_query("select * from tb_car_log where car_number='"$myrow[car_number]"); 
$myrows=mysql_fetch_array($querys); 
echo $myrows[car_log]; // 输 出 车 辆 的 使 用 日 志 信息 
?> 
</td> 
<td align="center" bgcolor="#FFFFFF"> 
<?php 
if($myrows[car_log]==null){ // 判 断 车 辆 日 志 是 否 为 空 ， 如 果 为 空 ， 则 执行 下 面 的 内 容 
echo "<a href='indexs.php?Imbs= 发 货 单 &ljid=$myrow[id]'> 使 用 该 车 </a>"; 
Jelse{ // 如 果 车 辆 日 志 不 为 空 ， 则 执行 下 面 的 内 容 
echo "<a href='indexs.php?Imbs= 发 货 单 &ljid=$myrow[id]'> 预 定 该 车 </a>"; 
全 
</td> 
</tr> 


<?php }}else{ ”echo "您 查找 的 路 线 不 存在 !"; }}?> 
< 代码 贴 十 
@ select2: 指定 到 达 地 点 。 


四 让 语句 : 判断 从 表单 中 获取 的 变量 是 否 为 真 。 
@ like: 执行 模糊 查询 ， 从 数据 库 中 读 取 符 合 条 件 的 数据 。 


5.6.3 ”车 源 信息 查询 模块 的 实现 过 程 


国 ”车 源 信息 查询 模块 使 用 的 数据 表 : tb_car、tb_car log 
车 源 信息 查询 模块 的 功能 是 根据 客户 提供 的 信息 ， 从 数据 库 中 读 取 有 关 某 个 路 线 的 车 辆 使 用 的 情 


Ci 
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形 ， 及 时 为 客户 确定 一 个 合理 的 物流 配送 方案 。 该 模块 的 运行 结果 如 图 5.24 所 示 。 
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物 订 下 关 仿 自 网 ， 信 浴 之 二 ， 序 力 打造 知 冯 品 竺 


车 源 信息 查询 功能 的 实现 通过 car_select.php 文件 来 完成 ， 首 先 与 数据 库 建 立 连 接 ， 然 后 创建 一 个 
form 表单 ， 将 客户 的 地 址 信息 提交 到 当前 页 ， 作 为 查询 的 条 件 ; 最 后 编写 PHP 脚本 语句 ， 以 表单 中 获 
取 的 地 点 数据 为 条 件 ， 执 行 like 模糊 查询 ， 从 数据 库 的 指定 表 中 查询 符合 条 件 的 信息 ， 并 且 将 查询 出 
的 信息 显示 到 页 面 中 。 其 关键 代码 可 以 参考 5.6.2 节 的 例 程 02。 


5.6.4 ”单元 测试 


在 对 车 源 信息 查询 模块 进行 测试 的 过 程 中 ， 当 输入 指定 的 出 发 点 和 到 达 地 点 进行 提交 后 ， 出 现 如 
图 5.25 所 示 的 页 面 ， 程 序 出 错 ， 没 有 找到 符合 条 件 的 记录 。 


| - 
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图 5.25 车 源 信息 查询 的 错误 运行 结果 


分 析 错 误 原 因 : 从 弹出 的 错误 信息 来 看 ， 错 误 应 该 与 数据 库 有 关 。 下 面 从 数据 库 开始 查找 错误 的 
原因 ， 首 先 确 定 程序 中 使 用 的 数据 库 、 数 据 表 和 字段 的 书写 是 否 正确 ， 仔 细 核 对 没有 发 现 问题 ， 然 后 
确认 使 用 的 select 查询 语句 的 格式 是 否 正确 , 经 检查 发 现 , 在 编写 select 查询 语句 时 使 用 了 错误 的 语句 
格式 , 在 执行 like 模糊 查询 时 没有 将 参数 的 两 侧 加 上 单 引号 (car_road like %$select1% and car_road like 
%S$select2%)， 从 而 导致 程序 中 类 型 不 匹配 的 错误 。 其 错误 语句 和 正确 语句 的 对 比如 图 5.26 所 示 。 


时 
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sult) >0) { 
fetch_array ($result)){ 


(au 
sult)>0) 
fetch_array ($result)) { 


图 5.26 错误 语句 和 正确 语句 对 比 


正确 的 格式 为 car road like '%S$select1%' and car road like '%$select2%'， 或 者 car road like 
'%".$select1."%' and car_road like '%".$select2."%'。 修 改 后 保存 程序 ， 程 序 正 常 运行 。 


ER 
应 该 加 上 单 引号 ， 表 示 该 数据 为 字符 型 ， 否 则 程序 将 发 生 类 型 不 匹配 的 错误 。 


/ 
说 在 PHP 中 的 字符 串 有 两 种 表示 格式 : 第 一 种 是 将 字符 串 用 单 引号 括 起 来 ; 第 二 种 是 将 字 
符 串 用 双 引 号 括 起 来 。 如 果 采 用 第 二 种 格式 ，PHP 变量 可 以 直接 写 在 该 字符 串 中 ， 系 统 不 会 将 该 变 
量 看 作 字 符 串 的 一 部 分 ， 而 是 将 其 看 作 PHP 的 合法 变量 。 


5.7 发 货 单 管理 模块 设计 


5.7.1 发 货 单 管理 模块 概述 


发 货 单 管理 模块 主要 包括 发 货 单 的 填写 、 发 货 单 查询 、 发 货 单打 印 和 发 货 单 删除 。 发 货 单 管理 模 
块 的 主要 功能 如 图 5.27 所 示 。 


管理 员 


图 5.27 ”发 货 单 管理 模块 的 功能 结构 图 


5.7.2 发 货 单 管理 模块 技术 分 析 


如 何 才能 发 挥 函 数 的 最 大 作用 ? 设计 者 不 但 要 能 够 从 表面 理解 函数 的 功能 和 作用 ， 而 且 要 能 够 以 
一 种 发 散 的 思维 方式 去 考虑 问题 ， 不 要 被 其 本 身 的 功能 所 束缚 ， 要 将 函数 中 某 些 功能 有 机 地 结合 起 来 ， 
从 而 发 挥 其 更 大 的 作用 。 
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在 发 货 单 管理 模块 中 应 用 到 一 个 发 货 单 的 编号 ， 该 编号 是 发 货 单 的 唯一 标识 ， 不 允许 存在 重复 。 
在 编写 生成 这 个 编号 的 程序 时 ， 首 选 方案 是 应 用 随机 函数 获取 随机 的 数字 作为 编号 ， 这 是 一 个 最 直接 
的 做 法 ， 但 是 考虑 到 发 货 单 编号 不 但 具有 唯一 性 ， 而 且 还 要 使 其 具有 一 定 的 标志 性 ， 从 这 个 角度 考虑 
使 用 随机 函数 就 有 些 不 适合 ， 因 为 其 不 具备 一 定 的 标志 性 。 这 时 ， 如 果 应 用 dateO0 函 数 ， 则 可 以 达到 上 
述 所 说 的 要 求 ， 不 但 编号 具有 唯一 性 ， 而 且 还 有 规律 可 循 。 

为 了 能 够 更 好 地 理解 这 种 发 散 思 维 的 编程 思想 ， 首 先 介绍 一 下 date() 函 数 。 语 法 格式 如 下 : 


string date ( string format , int timestamp) 


该 函数 返回 将 参数 timestamp 按照 指定 格式 格式 化 而 产生 的 字符 串 ,其 中 参数 timestamp 是 可 选 的 ， 
默认 值 为 time0， 即 如 果 没 有 给 出 时 间 戳 ， 则 使 用 本 地 当前 时 间 。 

date() 函 数 的 参数 format 的 格式 化 选项 如 表 5.1 所 示 。 

表 5.1 参数 format 的 格式 化 选项 
参数 说 了 明 

小 写 的 上 午 和 下 午 值 ， 返 回 值 为 am 或 pm 
大 写 的 上 午 和 下 午 值 ， 返 回 值 为 AM 或 PM 
Swatch Internet 标准 时 ， 返 回 值 为 000 一 999 
月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 为 01 一 31 
星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 为 Mon 一 Sun 
月 份 ， 完 整 的 文本 格式 ， 返 回 值 为 January 一 December 
小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 
小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 0 一 23 
小 时 ，12 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 01 一 12 
小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 00 一 23 
有 前 导 零 的 分 钟 数 ， 返 回 值 为 00 一 59 
判断 是 否 为 夏令 时 ， 如 果 是 ， 返 回 值 为 1， 否则 为 0 
月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 为 1 一 31 
星期 数 ， 完 整 的 文本 格式 ， 返 回 值 为 Sunday 到 Saturda 
判断 是 否 为 闵 年 ， 如 果 是 ， 返 回 值 为 1， 否则 为 0 
数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 为 01 一 12 
3 个 字母 缩写 表示 的 月 份 ， 返 回 值 为 Jan 一 Dec 
数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 
与 格林 威 治 时 间 相 差 的 小 时 数 ， 例 如 0200 
RFC 822 格式 的 日 期 ， 例 如 Thu，21 Dec 2000 16: 01: 07 +0200 
秒 数 ， 有 前 导 零 ， 返 回 值 为 00 一 59 
每 月 天 数 后 面 的 英文 后 级 ，2 个 字符 ， 例 如 st、nd、rd 或 者 th。 可 以 和 j 一 起 使 用 
指定 月 份 所 应 有 的 天 数 
本 机 所 在 的 时 区 
从 UNIX 纪元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 
星期 中 的 第 几 天 ， 数 字 表 示 ， 返 回 值 为 0 一 6 
ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 
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续 表 
说 了 明 
4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 
2 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
年 份 中 的 第 几 天 ， 返 回 值 为 0 一 366 


时 差 偏 移 量 的 秒 数 。UTC 西边 的 时 区 偏 移 量 总 是 负 的 ，UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 返 回 值 为 
-43200 一 43200 


和 6 注 总 这 里 有 效 的 时 间 蕉 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54 到 2038 年 1 月 


19 日 03:14:07( 此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 ) 在 Windows 系统 中 此 范围 限制 为 
从 1970 年 1 月 1 日 到 2038 年 1 月 19 日 。 


A 


[a 技巧 要 应 用 发 散 的 思维 方式 , 在 考虑 编程 方法 时 就 不 能 被 date() 函 数 自身 的 功能 所 限制 , date() 


函数 获取 的 时 间 不 只 可 以 用 来 记录 历史 中 的 菜 一 时 刻 ,而 且 可 以 将 其 应 用 到 其 他 的 地 方 ， 作 为 一 个 
有 规律 的 随机 数 。 


在 本 项 目 中 的 发 货 单 填 单 模块 中 就 是 应 用 date() 函 数 获取 当前 时 间 和 和 车辆 ID 作为 发 货 单 的 编号 。 
关键 代码 如 下 : 


<?php 
echo date("YmdHis"); // 获 取 系 统 的 当前 时 间 
echo $ljid; /获取 车 辆 的 ID 
是 
其 运行 结果 如 图 5.28 所 示 。 
语源 请 息 查询 MN sm: | 车 二 电话 : 
| RA! | xgAais:[ | txt: 厂 司 
国 洛 从 总 确认 EE | 
发 货 单 查 淘 货物 持 法 : 
等 忆 信息 管 理 习 
六 源 信 息 管理 习 
化 弛 攻 码 ts 加 


图 5.28 发 货 单 编号 生成 的 运行 结果 


AM 


[a 技巧 程序 开发 不 只 是 要 看 到 其 表面 的 功能 和 属性 ， 更 重要 的 是 将 某 些 功能 或 者 属性 有 机 地 结 
合 起 来 ， 从 而 开发 出 更 有 意义 的 功能 ， 这 才 是 程序 员 所 要 追求 的 。 例 如 ， 在 学 习 字 符 串 中 的 
str_ireplace() 函 数 时 ， 不 但 要 知道 该 函数 的 功能 是 实现 字符 串 的 替换 功能 ， 而 且 要 学 会 如 何 巧妙 地 
运用 其 实现 更 好 的 功能 。 当 换个 角度 去 考虑 这 个 函数 的 功能 时 ,对 这 个 函数 的 参数 字符 串 进 行 颜色 
改变 。 这 时 实现 的 功能 就 不 只 是 简单 的 字符 囊 蔡 换 ， 而 且 对 字符 囊 实现 了 描 红 的 功能 。 
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5.7.3 发 货 单 填 单 的 实现 过 程 


发 货 单 填 单 模块 使 用 的 数据 表 : tb_customer、tb_car log、tb_shopping 
发 货 单 的 填写 是 客户 在 确定 物流 配送 路 线 以 后 填写 的 一 个 货物 配送 详细 信息 单据 ， 内 容 包括 发 货 
单 编号 、 车 牌号 码 、 车 主 电话 、 发 货 人 、 发 货 人 电话 、 发 货 地 址 、 付 款 方式 、 货 物 描述 、 收 货 人 、 收 


货 人 电话 、 收 货 地 址 和 说 明 。 发 货 单 填 单 的 运行 结果 如 图 5.29 所 示 。 
Er ee re rz EF 
aA: eT [TY ey | 
区 大 天 下 E| 
EM 
于 
wm El 
则 
收 提 人 :了 甘 % ，，，，，，，，，EAkE: [5 
民有 国 
Ci 
到 
a 可 
Wp: 
到 


ES 
图 5.29 发 货 单 填 单 的 运行 结果 
发 货 单 填 单 的 实现 通过 insert_ dds.php 文件 来 完成 , 首先 随机 生成 一 个 订单 编号 ,然后 创建 一 个 form 
表单 ， 最 后 将 表单 中 的 数据 提交 到 数据 库 中 。 在 insert_dds.php 文件 中 使 用 的 重要 表单 元 素 如 表 5.2 
所 示 。 
表 5.2 发 货 单 填 单 中 使 用 的 重要 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 
method="post" action="insert_dd_ok.php" onSubmit="javascript: retum 

forml form 表单 
check ddO" 

fahuo id text id="fahuo id" value="<?php echo date("YmdHis"):?7><?php echo $ljid;?>” | 发 货 单 编 号 

car_ number text value="<?php echo $myrow[car number]?>" 车 牌号 码 

car tel text value="<?php echo $myrow [tel];?>" 车 主 电 话 
<option selected="selected"></option><option> 发 货 人 付款 </option> 

fahuo_ fk select 人 要 付款 方式 
<option> 第 三 方 付款 </option> 

car log textarea Cols="65" rows="5" 说 明 

Submit submit value=" 提 交 " 提交 表单 


将 表单 中 提交 的 数据 存储 到 数据 库 是 通过 insert_dd_ok.php 文件 来 完成 的 , 在 该 文件 中 首先 连接 数 
据 库 ,然后 通过 preg_match() 函 数 判断 表单 中 提交 的 电话 号 码 格式 是 否 正确 , 最 后 将 表单 中 提交 的 数据 
存储 到 tb_shopping 表 中 , 并且 将 车 辆 使 用 的 说 明 提 交 到 tb_car log 表 中 , 将 客户 信息 提交 到 tb_customer 
表 中 。 关 键 代 码 如 下 : 

倒 程 03 ”代码 位 置 ， 光盘 \TM\05\wlglinsert_dd_ok.php 
<? 
ee /| 判断 登录 的 用 户 是 否 正确 
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include("conn/conn.php"); /连接 数据 库 

$fahuo time=date("Y-m-d H:i:s"); /获取 系统 当前 时 间 

if($Submit==trueX{ // 如 果 提交 表单 ， 则 执行 下 面 的 语句 
eeeeeeeeteextwx 判 断 表 单 中 提交 的 电话 号 码 格 式 是 否 正确 ees 刘 1/ 


iipreg_match(VA(df3}-)Qdf8)3IA(d(4)-)Gdf795)SIACd(4-)Gdf8)SIACd(11)37,S$car_ tel$countes)i{ 
if(preg_match("/*(d{3})(\d{8})$I"Qd{4})(\d{7})SI"Qd{4})(\d{8})$I"Qd{11))$/", Sfahuo_tel,Scountes)X{ 


if(preg _， ee Oe Ct )(\d{8)$I^Qd{11})$/",$shouhuo_tel,$countes))}{ 

ests 将 表单 提交 的 数据 添加 到 tb_shopping 表 中 人 eer/ 

$query="insert into ee (car_number,car_tel,fahuo_id,fahuo_user, 
fahuo_tel,fahuo_address,fahuo_content,fahuo_time,fahuo_fk,shouhuo_user,shouhuo_address,shouhuo_tel)val 
ues('$car_number,'$car_tel','$fahuo_id','$fahuo_user','$fahuo_tel','$fahuo_address','$fahuo_content','$fahuo _ti 
me','$fahuo_fk','$shouhuo_user','$shouhuo_address','$shouhuo_tel)"; 

$result=mysql_query($query); 

六 weeeeeesssetssewetssy 将 车 辆 使 用 说 明 添加 到 tb_car_log 表 中 * eeeeeseeesreereeeeeees ol 

$querys="insert into 
tb_car_log(car_log,car_number,log_date,fahuo_id)values('$car_log','$car_number','$fahuo_time",'$fahuo_id")"; 

Sresults=mysql_query($querys); 

treets 将 客户 信息 提交 到 tb_customer 表 中 * wereseeeeseseeresreressets ol 

$queryss="insert into tb_customer 
(customer_user customer tel,customer_address)values($fahuo_user,'$fahuo_tel','$fahuo_address)"; 

Sresultss=mysql_query($queryss); 

echo "<script>alert(' 发 货 单 添加 成 功 !');window.location.href='indexs.php?Imbs= 发 货 单 ;</script>"; 

Jelse{ 
echo "<script>alert(' 您 输入 的 收 货 人 的 电话 号 码 格式 不 正确 !);history.back()</script>"; } 
}else{ 
echo "<script>alert(' 您 输入 的 发 货 人 的 电话 号 码 格式 不 正确 !!');history.back()</script>"; } 
}else{ 
echo "<script>alert( 您 输入 的 车 主 的 电话 号 码 格式 不 正确 山 );history.back()</script>"; } 

}?> 
<?php jelse{ // 判 断 用 户 是 否 正确 登录 ， 如 果 不 是 则 返回 登录 页 面 

echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>";} 
?> 


5.7.4 发 货 单 查询 的 实现 过 程 


国 发 货 单 查询 模块 使 用 的 数据 表 : tb_car log、tb shopping 

发 货 单 查询 是 为 了 便于 对 发 货 单 进行 查找 
以 及 处 理 而 设计 的 一 个 功能 ， 通 过 其 可 以 准确 ”下 
地 查找 到 指定 的 发 货 单 ， 并 且 还 设置 一 个 发 货 -ss 后 
单 删除 的 功能 ， 可 以 对 已 经 失效 或 者 作废 的 发 。 ee | 
货 单 进行 删除 。 其 实现 的 原理 主要 是 通过 以 发 [| rr i 
货 单 编号 为 条 件 的 精确 查找 或 者 以 发 仙人 姓名 sos 了 车 
为 条 件 的 模糊 查询 ， 然 后 将 查询 的 结果 输出 到 。 sw 二 
页 面 中 。 运 行 结果 如 图 5.30 所 示 。 全 ar 

发 货 单 查询 的 实现 主要 通过 hwys.php 文 件 图 530 ”发 货 单 查询 的 运行 结果 


全 


Er er Me | 
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完成 ， 首 先 与 数据 库 建立 连接 ， 然 后 创建 一 个 表单 forml， 通过 该 表单 来 提交 查询 的 条 件 ， 可 以 选择 精 
确 查找 (以 发 货 单 编号 为 条 件 ) 或 者 模糊 查询 (以 发 货 人 姓名 为 条 件 )， 最 后 根据 form1 表单 中 提交 的 
数据 ， 执 行 查询 语句 ， 从 数据 库 中 读 取出 符合 条 件 的 发 货 单 的 内 容 。 程 序 的 关键 代码 如 下 : 


作 。 


全 性 04 ”代码 位 置 : 光盘 VTMWS\wlgIihwysphp 


<?php session_start(); include("conn/conn.php"); /| 初始 化 session 变量 ， 连 接 数 据 库 
if($_SESSION[admin_user]==trueX{ // 判 断 管理 员 是 否 是 正确 登录 

?> 

<!-- 省 略 了 部 分 代码 --> 

<?php 


广 *esevssesstets 判 断 当 form1 表单 提交 的 值 为 发 货 单 编号 时 ， 执 行 下 面 的 语句 * see 刘 
if($select1==" 发 货 单 编号 " and $select!l=""X{ 
$query="select * from tb_shopping where fahuo _id='$select"; 


$result=mysql_query($query); /执行 查询 语句 
$myrow=mysql_fetch_array($result); /获取 查询 结果 


oreseesesttwsttt 当 form1 表单 提交 的 值 为 发 货 人 时 ， 执 行 下 面 的 语句 * eseaexseewsesr ol 
if($select1==" 发 货 人 " and $select!=""X{ 
$query="select * from tb_shopping where fahuo_user='$select"; 


$result=mysql_query($query); /检索 指定 发 货 人 的 信息 
$myrow=mysql_fetch_array($result); 
吃 
wa /省 略 部 分 HTML 代码 
<td colspan="4" bgcolor="#FFFFFF"><textarea name="car_log"> 
< 一 - 一 一 一 一 一 一 一 一 一 输出 车 辆 日 志 中 的 车 辆 使 用 情况 一 一 一 一 一 -一 一 一 一 一 -一 一 
<?php 
$querys="select* from tb_car_log where fahuo_id='$myrow[fahuo_idj"; // 从 车 辆 日 志 表 中 查找 指定 的 数据 
$results=mysql_query($querys); /执行 查询 语句 
$myrows=mysql_fetch_array($results); /获取 查询 结果 
echo $myrows[car_log]; // 输 出 车 辆 日 志 信 息 
人 
</textarea></td> 
< 一 一 根据 发 货 单 的 1D 设置 一 个 发 货 单 删除 的 超 链接 -一 一 一 一 一 一 一 一 一 一 一 一 一 - -- > 
<td bgcolor="#FFFFFF"><a href="fhd_qr.php?delete=<?php echo $myrowfid];?>"> 删 除 发 货 单 </a></td> 
/省 略 部 分 HTML 代码 


<?php }else{ ”// 判 断 管理 员 是 否 正确 登录 ， 如 果 错 误 则 返回 到 登录 页 
echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>"; 
} 


?> 
发 货 单 删除 的 操作 通过 fhd_qrphp 文件 来 完成 ,主要 根据 超 链接 中 提供 的 发 货 单 ID 执行 删除 的 操 
关键 代码 如 下 : 

倒 程 05 ”代码 位 置 : 光盘 \TM\05\wlglMhd_gr.php 


<?php session_start(); include("conn/conn.php"); /| 初始 化 session 变量 ， 连 接 数 据 库 
if($_SESSION[admin_user]==true}{ 
if($delete==true}{ // 判 断 提 交 的 发 货 单 ID 是 否 为 真 


$query="delete from tb_shopping where id='$delete'"; 。”// 根 据 发 货 单 ID 执行 删除 发 货 单 的 操作 
$result=mysql_query($query); 
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echo "<script>alert(' 发 货 单 删 除 成 功 !");window.location.href='indexs.php?Imbs= 发 货 单 查询 ,</script>"; 


en 发 货 单 删除 成 功 ， 弹 出 提示 ， 并 返回 到 上 一 页 eeeerex* 刘 
} 
}else{ 
ee 发 货 单 删除 失败 ， 弹 出 提示 ， 并 返回 到 上 一 页 eeee* tl 
echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>"; 
} 
i 


5.7.5 发 货 单打 印 的 实现 过 程 


发 货 单 的 打印 是 为 货物 的 发 送 提供 一 个 书面 的 依据 , 作为 
企业 与 客户 商业 活动 的 凭证 ， 其 内 容 必 须 真实 、 准 确 。 发 货 单 
打印 的 运行 结果 如 图 5.31 所 示 。 

这 里 的 发 货 单 打印 技术 主要 通过 框架 打印 技术 来 完成 , 即 
只 打印 框架 中 的 内 容 , 而 网 页 中 其 他 的 内 容 不 打印 。 实现 原理 
是 : 首先 在 insert_dd.php 文件 中 创建 一 个 浮动 框架 , 设置 要 指 
定 打印 内 容 的 范围 , 并 且 连 接 到 要 打印 的 insert_dds.php 文件 ， 
然后 在 指定 的 文件 中 输出 要 打印 的 内 容 ， 最 后 应 用 onClick 事件 
调用 parent.content.focus() 和 window.print0 方 法 实现 打印 功能 。 

在 insert_dd.php 文件 中 创建 一 个 浮动 框架 , 设置 框架 名 称 
为 content， 设 置 打印 的 范围 宽 97%、 高 232%， 连 接 到 要 打印 
的 insert_dds.php 文件 ， 程 序 代码 如 下 : 

倒 程 06 ”代码 位 置 ， 光盘 \TM\05\wlghinsert_dd.php 


EEE EE 
党 现 | 过 硕 | 
这 后 打 机 
2 包 
六 量 ] 
未 加 了 印 机 。 Epton | 
[ee 
姜 中 i 
Ea ETT 
厂 打 fp 列 文件 ) 
Bt 
全 部 四 EE [m= 
人 巡 示 区 轩 四 ”人 当 阁 观 而 信 
WHO jy 
Ne EEC | Son DIDdDH 
La 原 用 节 ) 


5.31 发 货 单打 印 的 运行 结果 


<iframe name="content" src="insert_dds.php?ljid=<?php echo S$ljid;?>" frameborder="0" width="97%" 


height="252%"></iframe> 


设置 要 打印 的 内 容 ， 通 过 insert_dds.php 文件 来 完成 ， 发 货 单 填写 的 内 容 以 表单 的 形式 显示 ， 其 中 
的 内 容 可 以 参考 光盘 中 的 TM\05Wwlglinsert_dds.php 文件 ， 这 里 不 再 袭 述 。 
最 后 回 到 insert_dd.php 页 中 ,在 该 页 中 设置 一 个 超 链接 , 应 用 onClick 事件 调用 parent,content.fbcus0 


和 window.print() 方 法 实现 发 货 单 的 打印 功能 。 程 序 关键 代码 如 下 : 


倒 程 07 ”代码 位 置 ， 光 盘 \TM\05\wlghinsert_dd.php 
<a href="#" 
‘onClick="parent.content.focus();window.print();"> 


<img src="images/d!_033.gif" width="87" height="27" border="0"> 


</a> 


5.7.6 单元 测试 


在 进行 发 货 单 填 单 模块 的 测试 过 程 中 ， 当 打开 发 货 单 填 单 模块 的 页 面 时 ， 其 运行 结果 如 图 5.32 


所 示 。 
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在 该 页 面 中 整个 浮动 框架 中 的 内 容 没 有 显示 。 分 析 错 误 原因 ， 应 该 是 在 设置 浮动 框架 时 出 现 的 问 
题 ， 查 看 insert_dd.php 文件 中 的 代码 ， 发 现 原来 是 设置 浮动 框架 时 指定 的 链接 文件 出 现 问题 ， 使 用 了 
错误 的 链接 文件 名 称 。 链 接 文 件 的 正确 名 称 是 insert_dds.php， 而 程序 中 将 该 文件 的 名 称 书写 为 inser_ 
ddes.php， 从 而 导致 页 面 中 没有 输出 任何 的 内 容 。 将 insert_dd.php 文件 中 的 内 容 进行 修改 ， 重 新 运行 程 
序 ， 运 行 结果 如 图 5.33 所 示 。 
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图 5.32 发 货 单 填 单 模 块 的 运行 结果 图 5.33 发 货 单 模块 的 运行 结果 


5.8 回执 单 验收 管理 模块 设计 


5.8.1 回执 单 模块 概述 


回执 单 模块 的 主要 功能 就 是 对 货物 配送 完成 确认 : 将 该 发 货 单 执行 类 型 更 新 为 “1”， 表 明 该 次 物 
流 配送 已 经 完成 ， 清 空 该 车 辆 的 使 用 日 志 ， 便 于 执行 下 一 个 订单 。 该 模块 的 业务 操作 流程 如 图 5.34 
所 示 。 


eh 回执 单 


图 5.34 ”回执 单 模块 的 业务 操作 流程 


5.8.2 ”回执 单 模 块 技术 分 析 


在 回执 单 模块 中 ， 要 对 回执 单 进行 确认 首先 要 查找 到 指定 的 发 货 单 ， 并 且 对 其 内 容 进 行 核实 。 这 
里 应 用 的 是 根据 发 货 单 的 编号 进行 查询 ， 从 数据 库 中 读 取出 符合 条 件 的 数据 。 在 执行 查询 语句 、 从 数 
据 库 中 读 取 数据 时 ， 主 要 应 用 mysql_query0、mysql_num_rows0 和 mysql_fetch_array0) 函 数 。 

首先 通过 mysql_query0) 函 数 执行 一 个 select 查询 语句 ， 然 后 通过 mysql_num_rows(O) 函 数 判断 查询 
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的 结果 是 否 有 值 ， 如 果 没 有 值 则 输出 “您 查找 的 发 货 单 编号 不 存在 !”， 如 果 有 值 则 执行 mysql_fetch_ 
arrau(0) 函 数 ， 输 出 数据 库 中 查询 到 的 数据 。 

mysql_queryO 函 数 向 与 指定 的 连接 标识 符 关 联 的 服务 器 中 的 当前 活动 数据 库 发 送 一 条 MySQL 查 
询 。 语 法 如 下 : 

resource mysql_query ( string query [, resource link_identifier] ) 

mysql_query() 函 数 的 参数 query 是 字符 串 的 类 型 ， 传 入 的 是 SQL 指令 ;参数 link_identfier 是 资源 
类 型 ， 传 入 的 是 由 mysql_connect(O) 函 数 或 者 mysql_pconnect(0) 函 数 返 回 的 连接 号 。 如 果 省 略 该 参数 ， 则 
会 使 用 最 后 一 个 打开 的 MySQL 数据 库 连 接 。 
和 注意 去 询 字符 中 不 应 以 分 号 未， 

mysql_num_rows() 函 数 获取 结果 集中 行 的 数目 。 语 法 如 下 : 

int mysql_num_rows ( resource result ) 


《注意 该 函数 仅 对 SELECT 语句 有 效 。 


mysql_fetch_array() 函 数 返回 根据 从 结果 集 获取 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false。 语 
法 如 下 : 


array mysql_fetch_array ( resource result [, int result_type] ) 


mysql_fetch_array() 函 数 的 参数 说 明 如 表 5.3 所 示 。 
表 5.3 ”mysql_fetch_array() 函 数 的 参数 说 明 
参数 名 称 


result 


资源 类 型 的 参数 ， 要 传 入 的 是 由 mysql_query0 函 数 返回 的 “数据 指针 ” 

可 选项 ， 整 数 型 参数 ， 要 传 入 的 是 MYSQL_ASSOC、MYSQL NUM 和 MYSQL_BOTH 3 种 由 PHP 
定义 好 的 常数 之 一 ， 默 认 值 是 MYSQL_BOTH 

用 MYSQL_ASSOC 只 得 到 关联 索引 (相当 于 mysql_fetch_assoc() 函 数 ) 

用 MYSQL_NUM 只 得 到 数字 索引 (相当 于 mysql_fetch_row() 函 数 ) 

用 MYSQL_BOTH 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 


result_type 


和 注意 该 函数 返回 的 字段 名 是 区 分 大 小 写 的 。 
通过 上 述 3 个 函数 就 可 以 完成 从 数据 库 中 读 取 数据 ， 并 且 将 数据 输出 到 页 面 的 功能 。 


po 
据 库 中 所 有 符合 条 件数 据 的 循环 输出 。 


@ 
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应 用 while 循环 语句 读 取 数 据 库 中 的 数据 , 这 里 以 读 取 tb_shopping 表 中 的 数据 为 例 , 程序 代码 如 下 : 


<?php include("conn/conn.php"); /| 连接 数据 库 
$query="select * from tb_shopping "; /| 编写 查询 语句 
$result=mysql_query($query); /| 执行 查询 语句 

@ if(mysql_num_rows($result)<1X{ 1/ 判断 数据 库 中 是 否 存 在 数据 
echo "您 查找 的 内 容 不 存在 1"; 
jelse{ 

@ while($myrow=mysql_fetch_array(S$result)X{ // 循 环 输出 数据 库 中 的 数据 

?> 

<?php 

echo $myrow[id]; // 输 出 数据 库 中 的 数据 

9% 

// 输 出 内 容 的 代码 部 分 略 

<?php }?> 

Ah 代码 贴 二 


@ mysql_num_rows0: 返回 根据 从 结果 集 获 取 的 行 生成 的 数组 ， 如 果 没有 更 多 行 则 返回 false。 
@ mysql_fetch_array(): 获取 结果 集中 行 的 数目 。 该 函数 只 对 SELECT 语句 有 效 。 


5.8.3 ”回执 单 验收 模块 的 实现 过 程 


国 ”回执 单 验收 模块 使 用 的 数据 表 : tb_car log、tb_shopping 
回执 单 验 收 模块 的 主要 功能 就 是 对 货物 配送 完成 确认 ， 运 行 结果 如 图 5.35 所 示 。 
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5.35 ”回执 单 验收 模块 的 运行 结果 


回执 单 验收 模块 的 实现 过 程 是 ;首先 在 select_dhd.php 页 中 根据 发 货 单 的 编号 查询 出 指定 的 发 货 
单 ， 对 发 货 单 进行 核实 ， 然后 单 击 “ 发 货 订单 回执 确认 ” 超 链接 ， 实 现 回 执 单 的 验收 。 关 键 代码 如 下 : 
倒 程 08 ”代码 位 置 ， 光盘 \TM\05\wlgl\select_dhd.php 
<?php 
include("conn/conn.php"); // 连 接 数据 库 
if($Submit==true){ 
$query="select * from tb_shopping where fahuo_id='$select"; // 编 写 查询 语句 
$result=mysql_query($query); /执行 查询 语句 
if(mysql_num_rows($result)<1){ // 判 断 数 据 库 中 是 否 存在 数据 


@ 
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echo "您 查找 的 发 货 单 编号 不 存在 "; 
Jelse{ 
$myrow=mysql_fetch_array($result); /| 输 出 数据 库 中 的 数据 
ek 
/| 输出 内 容 的 代码 部 分 略 
<td colspan="4" bgcolor="#FFFFFF"> 
<textarea name="car_log"> 
<?php 
大 oo 根据 提交 的 货物 编号 ， 查 询 车 辆 日 志 中 的 对 应 日 志 * rr 
$querys="select * from tb_car log where fahuo_id='$myrow[fahuo_id]” 
$results=mysql_query($querys); /执行 查询 语句 
$myrows=mysql_fetch_array($results); /| 输出 数据 库 中 的 数据 
echo $myrows[car_log]; /| 输出 数据 库 中 的 数据 
?> 
</textarea></td> 
// 输 出 内 容 的 代码 部 分 略 
< 一 一 一 一 一 一 一 一 一 设置 超 链接 ， 执 行 发 货 单 的 确认 -一 -一 一 -一 -一 一 -> 
<td bgcolor= 


#FFFFFF"><a href="fhd_qr.php?fahuo_id=<?php echo $myrow[fahuo_id];?>"> 发 货 订单 回执 
确认 </a></td> 


最 后 在 fhd_qr.php 页 根据 超 链 接 提 供 的 发 货 单 编号 更 新 发 货 单 类 型 为 1, 清 空 对 应 车 辆 的 使 用 日 志 
程序 的 关键 代码 如 下 : 


倒 程 09 ”代码 位 置 ， 光盘 \TM\05\wlgl\Mfhd_qrphp 


<?php session_start(); include("conn/conn.php"); /初始 化 session 变量 
if($_SESSION[admin_user]==true){ // 判 断 用 户 是 否 是 正确 登录 
if($_GETIfahuo_id]==trueX{( 


// 判 断 超 链接 传递 的 变量 值 是 否 为 真 
1* vexesxxxwrwxkexx 以 超 链接 传递 的 变量 值 为 条 件 ， 编 写 删 除 语句 *e*sexexsexexexeeex xl 
$query="delete from tb_car_log where fahuo_id='$fahuo_id"; 
$result=mysql_query($query); // 执 行 删除 语句 
Je sexesxxwxwxsexwswts 更 新 发 货 单数 据 表 中 发 货 的 类 型 为 T*sssssseesreesxsexessy of 
$query="update tb_shopping set fahuo_ys='1' where fahuo_id='$fahuo_id' "; 
$result=mysql_query($query); 
echo "<script>alert(' 发 货 回执 单 处 理 成 功 !");window.location.href='indexs.php?Imbs= 回 执 发 货 单 
确认 ';</script>"; } 
Jelse{ 
echo "<script>alert(' 请 您 正确 登录 !"); window .location.href='index.php';</script>" 
} 


?> 


5.9 基础 信息 管理 模块 设计 
5.9.1 基础 信息 管理 模块 概述 


基础 信息 管理 模块 主要 包括 客户 信息 管理 ， 用 于 添加 客户 信息 和 删除 客户 信息 ; 车 源 信息 管理 ， 
主要 用 于 对 车 源 信息 进行 管理 ， 实 现 车 源 信息 的 添加 、 修 改 和 删除 功能 ;管理 员 信息 管理 ， 用 于 修改 
管理 员 的 密码 。 基 础 信息 管理 模块 的 功能 结果 如 图 5.36 所 示 


@ 
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图 5.36 ”基础 信息 管理 模块 的 功能 结构 图 


5.9.2 ”管理 员 信 息 管理 模块 技术 分 析 


修改 管理 员 密码 的 模块 中 ， 在 对 密码 进行 处 理 的 过 程 中 用 的 是 面向 对 象 技术 实现 密码 的 修改 。 
面向 对 象 技术 的 关键 就 是 类 的 创建 ， 首 先 创建 一 个 chkuser 类 ， 其 中 定义 5 个 数据 成 员 ， 实 现 对 管 
理 员 密码 的 修改 ， 分 别 是 管理 员 名称 〈$admin_user)、 管 理 员 密码 ($admin_pass)、 管 理 员 新 密码 
($admin_new_pass)、 管 理 员 新 密码 确认 ($admin_new_pass2) 和 验证 码 ($yzm)， 然 后 应 用 chkuser0 
函数 实现 数据 成 员 的 初始 化 ， 最 后 应 用 数据 成 员 函 数 chkinput0 〇 实现 密码 的 修改 。 程 序 代 码 如 下 : 
倒 程 10 ”代码 位 置 ， 光盘 \TM\05\wlgl\update_pass_ok.php 


<?php session_start(); // 初 始 化 session 变量 

@ classchkuser{ // 创 建 一 个 chkuser 类 
var $admin_user; // 创 建 数据 成 员 ， 用 户 名 
var $admin_pass; // 创 建 数据 成 员 ， 密 码 
var $admin_new_pass; // 创 建 数据 成 员 ， 新 密码 
var $admin_new_pass2; /创建 数据 成 员 ， 密 码 确认 
var $yzm; /创建 数据 成 员 ， 验 证 码 

function chkuser($x,$y$m,$n,$zj{ /创建 chkuser() 函 数 ， 实 现 数据 成 员 初始 化 


S$this->admin_user=$x; 
S$this->admin_pass=$y; 
S$this->admin_new_pass=$m; 
S$this->admin_new_pass2=$n; 
S$this->yzm=$z; 


} 
@ function chkinput(){ // 创 建 chkinput() 函 数 ， 实 现 密码 更 改 
// 判 断 输入 的 验证 码 是 否 正确 
© if(strval($this->yzm)!=$_SESSION["autonum1"]X{ 
echo "<script>alert( 验 证 码 输入 错误 !");history.back();</script>"; 
exit; 


} 
include("conn/conn.php"); /连接 数据 库 
// 判 断 输 入 的 密码 是 否 正确 


$query=mysql_query("select * from tb_admin where admin_user=".$this->admin_user." and 


@ 
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admin_pass=".$this->admin_pass.""); 
if(mysql_num_rows(S$query)<1X{ 


fst *** 如 果 密码 不 正确 ， 则 提示 重新 输入 密码 eet . 
echo "<script>alert(' 您 输入 的 密码 不 正确 ， 请 重新 输入 ');history.back();</script>"; 
}else{ 


六 oo 如 果 密 码 正确 ， 则 应 用 更 新 语句 ， 实 现 密码 的 更 改 和 和 rr 
$query1=mysql_query("update tb_admin set admin_pass=".$this->admin_new_pass.""); 
if($query1==trueX{ 

echo "<script>alert(' 密 码 更 改 成 功 ");history.back();</script>"; 
} 


} 
} 


r ss bi 
®@ schk=new 
chkuser($_POST[admin_user],md5($_POST[admin_pass]),md5($_POST[admin_new_pass]),$_POST[admin_ 
new_pass2],$_POST[yzm]); 

S$chk->chkinput(); 
?> 


< 代码 贴 二 
@ chkuser: 创建 一 个 类 名 称 为 chkuser， 用 于 修改 管理 员 密 码 。 
@ chkinput 0: 创建 一 个 自 定义 函 数 chkinput()， 用 于 实现 密码 修改 。 
@ strval (): 获取 变量 的 字符 囊 。 
@ new chkuser: 实现 类 的 实例 化 ， 应 用 类 实现 密码 修改 。 


5.9.3 客户 信息 管理 的 实现 过 程 


国 客户 信息 管理 模块 使 用 的 数据 表 : tb_customer 
客户 信息 管理 模块 的 主要 功能 就 是 向 数据 库 中 添加 客户 信息 ， 并 且 可 以 对 指定 的 客户 进行 删除 操 
作 。 客 户 信息 管理 模块 的 运行 结果 如 图 5.37 所 示 。 


ET ET 


EPRS: | 


下 系 地 直 jy 习 
EI 
PES | [3 联 示 地 性 
ee ET 二 


旱 叶 区 


EE | 90604930734 WF 


5.37 客户 信息 管理 模块 的 运行 结果 


客户 信息 管理 模块 主要 通过 customerphp、customer_ok.php 和 delete_customerphp 文件 来 完成 。 首 
先 通过 customerphp 文件 来 添加 客户 信息 ， 并 且 输 出 数据 库 中 的 所 有 客户 信息 ， 设 置 删除 客户 信息 的 
超 链接 。 其 输出 客户 详细 信息 的 关键 代码 如 下 : 

倒 程 11 ”代码 位 置 : 光盘 \TM\05\wlgl\customerphp 


全 
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<?php include("conn/conn.php"); /连接 数据 库 
$query=mysql_query("select * from tb_customer "); /执行 查询 语句 ， 从 数据 库 中 读 取 数据 
while($myrow=mysql_fetch_array($query)){f // 通 过 while 循环 语句 ， 循 环 输出 数据 库 中 的 数据 
?> 
<tr> 


<td height="26" align="center" bgcolor="#FFFFFF"><?php echo $myrow[customer_user];?></td> 

<td align="center bgcolor="#FFFFFF"><?php echo $myrow[customer_tel];?></td> 

<td align="center" bgcolor="#FFFFFF"><?php echo $myrow[customer_address];?></td> 

<!-- 一 一 一 一 设置 删除 客户 信息 的 超 链接 ， 并 且 设 置 客户 的 1D 为 变量 -一 -一 一 -一 一 一 -> 

<td align="center bgcolor="#FFFFFF"><a href="delete_customer.php?delete=<?php echo 
$myrow[customer_id];?>"> 删 除 </a></td> 
</tr> 

<?php}?> 


客户 信息 添加 的 操作 通过 customer_ok.php 文件 完成 ， 首 先 连接 数据 库 ， 然 后 判断 提交 的 电话 号 码 
格式 是 和 否 正确 ， 最 后 将 客户 信息 数据 提交 到 数据 库 中 。 程 序 的 代码 如 下 : 
倒 程 12 ”代码 位 置 ， 光盘 \TM\05\wlgl\customer_ok.php 


<?php include("conn/conn.php"); /| 连接 数据 库 
if($Submit==true and $customer_user==trueX{ /判断 提交 的 值 是 否 为 真 
* osxsexsresrexwtttxwrwstst 判 断 提交 的 电话 号 码 格式 是 否 正确 习 srssrsrswessessrsreswoss ol 


if(preg_match("/*(\d{3}-)(\d{8})$I^Qd{4}-)(\d{7})$I^(\d{4}-)(\d{8})$I^(\d{11})$/",$customer_tel,$countes)X{ 
eeeesrtsestittsthrywy 将 提交 的 数据 添加 到 客户 信息 表 中 sevxeeeeeresrereseerx 训 
$query=mysql_query("insert into tb_customer(customer_usercustomer_ tel, customer_address) 
values($customer_user,$customer tel,$customer_address')); 
if($query==true){ 1/ 如果 添加 操作 成 功 ， 弹 出 提示 ， 并 返回 首页 
echo "<script> alert(' 客 户 信息 添加 成 功 ');window.location.href='indexs.php?Imbs= 客 户 信息 
管理 ';</script>"; 


} 
Jelse{ // 如 果 添 加 操作 失败 ， 弹 出 提示 ， 并 返回 上 一 页 
echo "<script>alert(' 您 输入 的 电话 号 码 格式 不 正确 !):history.back()</script>"; 
} 
} 
ns 


客户 信息 删除 的 操作 通过 delete_customer.php 文件 来 完成 ， 主 要 以 超 链接 中 提交 的 变量 为 根据 ， 删 
除数 据 库 中 指定 的 客户 信息 。 程 序 的 关键 代码 如 下 : 
倒 程 13 ”代码 位 置 ， 光盘 \TM\05\wlgl\delete_customer.php 


<?php include("conn/conn.php"); /连接 数据 库 
if($delete==true}{ // 判 断 提交 的 变量 值 是 否 为 真 
ss 根据 $ GET 获取 的 变量 值 为 依据 ， 执 行 删除 语句 季 和 were sl/ 
$query=mysql_query("delete from tb_customer where customer_id='$_GET[delete]"); 
if($queryX{ 1/ 如 果 删 除 操作 成 功 ， 则 弹出 提示 ， 并 返回 首页 
echo "<script> alert(' 客 户 信息 添加 成 功 );window.location.href='indexs.php?lImbs= 客 户 信息 管理 
"</script>"; 
} 
} 


?> 


多 
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5.9.4 车 源 信息 管理 的 实现 过 程 


国 ”车 源 信息 管理 模块 使 用 的 数据 表 : tb_car 


车 源 信息 管理 模块 主要 实现 对 车 辆 信息 的 添加 、 修 改 和 删除 操作 。 车 源 信息 管 理 模块 的 运行 结 


如 图 5.38 所 示 。 
车 源 信息 查 淘 
E27 IE 
书坊 发 贷 单 确认 
5 #5: | EE 
二 7 | 
BSS | pi | 
二 :从 EE 
LW | 9 
到 | | | 
图 5.38 车 源 信息 管理 模块 的 运行 结果 
车 源 信息 管理 模块 主要 由 两 个 文件 组 成 。 一 个 是 car.php 文件 ， 以 表单 文件 为 3 


洁 果 


E， 用 于 输出 车 辆 的 


信息 和 提交 车 辆 的 信息 。 设 置 form2 表单 ， 根 据 车 牌号 码 从 数据 库 中 读 取 对 应 车 辆 的 信息 ， 进 而 实现 


对 该 车 辆 信息 的 修改 或 者 删除 操作 ; 


加 车 辆 信息 到 数据 库 。forml 表单 中 使 用 的 重要 元 素 如 表 5.4 所 示 。 
表 5.4 ”车 源 信息 管理 模块 中 使 用 的 重要 表单 元 素 


设置 forml 表单 ， 用 于 显示 从 数据 库 中 读 取 的 车 辆 信息 和 直接 添 


名 称 元 素 类 型 重要 属性 含义 
go nis wo Poet 人 表单 
onSubmit="javascript:return check_car(form1):" 
username text id="username" value="<?php echo Smyrows[usernamel:2>” 车 主 姓名 
car_number text id="car_number" value="<?php echo $myrows[car_number];?>" 车 牌号 码 
car tel text value="<?php echo $myrowltel]:?>" 车 主 电话 
user_ number text id="user number" value="<?php echo $myrows[user number]:?>" 身份 证 号 码 
car_content textarea <?php echo $myrows[car_content]:?> 车 辆 描述 
address textarea <?php echo $myrows[address]:?> 家 庭 地 址 
car road textarea <?php echo $myrows[car_road];?> 运输 路 线 
Submit submit value=" 提 交 " 提交 表单 
Submit2 submit value=" 修 改 " 修改 表单 
Submit4 submit value=" 删 除 " 删除 表单 
另 一 个 是 car_insert_ok.php 文件 ,通过 该 文件 实现 对 表单 中 提交 的 数据 进行 处 理 。 首 先 连 接 数据 库 ， 
然后 根据 表单 中 提交 的 值 进行 判断 ， 当 按钮 SSubmit 的 值 为 “提交 ”时 ， 将 提交 的 数据 存储 到 指定 的 数 


据 表 中 ; 当 按 钮 SSubmit2 的 值 为 “修改 ”时 ， 则 更 新 话 


的 值 为 “删除 ” 则 执行 删除 语句 ， 将 指定 的 车 辆 信息 删除 。 程 序 关键 代码 如 下 : 


@ 


时 ， 


消 句 ， 对 车 辆 的 信息 进行 更 新 ; 当 按钮 $Submit4 
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倒 程 14 ”代码 位 置 : 光盘 \TM\05\wlgl\car_insert_ok.php 


<?php session_start(); include("conn/conn.php"); // 初 始 化 session 变量 ， 连 接 数据 库 
@ if($_SESSION[admin_user]==trueX{ // 判 断 管理 员 是 否 是 正常 登录 
if($Submit==" 提 交 "X // 判 断 表单 提交 的 值 是 否 为 “提交 ” 
广 eeeeeeeeeex* 判 断 表单 中 提交 的 电话 号 码 格 式 是 否 正确 et** 
四 if(preg_match("Ad{17}N\dIXII\d{15}/",$user_number,$counts)X{ 


if(preg_match("/^(\d{3}-)(\d{8})$I^(\d{4}-)(d{7})$I^(\d{4}-)(\d{8})$I^(\d{11})$/", $user_tel,$countes)\{ 
ee 将 表单 中 的 数据 添加 到 tb_car 数据 表 中 下 ee* */ 
$query=mysql_query("insert into tb_car(usemame,user_number,tel,address,car_number, car_road, 
car_ content) values('$usemame','$user_number','$user_tel','$address','$car_number','$car_road','$car_content)"); 
if($query==trueX{ 
echo "<script>alert(' 车 源 信息 添加 成 功 !");window.location.href='indexs.php?Imbs= 车 
源 信息 管理 ';</script>"; } 


Jelse{ 
echo "<script>alert(' 您 输入 的 电话 号 码 格式 不 正确 !!);history.back()</script>";。} 
Jelse{ 
echo "<script>alert(' 您 输入 的 身份 证 号 码 的 格式 不 正确 路 );history.back()</script>"; } 
} 
1 osreesveeseeesreeseeesveests 判 断 表 单 提交 的 值 是 否 为 “修改 ”vveeseeeseeeseessressress 吉 
[3 if($Submit2==" 修 改 "){ 


电话 号 码 的 格式 是 否 正确 和 和 srt 
if(preg_match("Ad{17}N\d|X]I\d{15}/",$user_number,$counts)}X{ 
if(preg_match("/^(\d{3}-)(\d{8})$I*(d{4}-)(d{7})S$I‘Nd{4}-)(d{8)$ Nd{11})$/", Suser_tel,Scountes)X{ 
et 新 tb_car 数据 表 中 的 数据 *ewerrsreeeeseseeseseewesws 
$query="update tb_car set username='$usermame', user_number='$user_number', tel='$user_tel', 
address='$address', car_number='$car_number, car_road='$car_road', car_content='$car_content"; 
$result=mysql_query($query); 
if($result==trueX{ 
echo "<script>alert(' 车 源 数 据 更 新 成 功 !!");window.location.href='indexs.php?Imbs= 车 
源 信 息 管理 ';</script>";} 
Jelse{ 
echo "<script>alert(' 您 输入 的 电话 号 码 格式 不 正确 由 );history.back()</script>";} 
Jelse{ 
echo "<script>alert(' 您 输入 的 身份 证 号 码 的 格式 不 正确 !");history.back()</script>"， } 


} 
ntti 单 提交 的 值 是 否 为 “删除 ”**eee 和 ttn 
0 if($Submit4==" 删 除 "){ 
$query="delete from tb_car where car_number='$car_number"; // 删 除 tb_car 数据 表 中 的 数据 
$result=mysql_query($query); 
if($result==trueX{ 
echo "<script>alert(' 车 源 数 据 删 除 成 功 !");window.location.href='indexs.php?Imbs= 车 
源 信息 管理 ,</script>"; 


} 
3% 
<?php }else{ 
echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>"; 
} 
?> 
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二 代码 贴 十 

@S$_ SESSION[admin user]: 判断 用 户 是 否 是 正常 登录 。 

@ preg_match(): 应 用 preg_match 验证 电话 号 码 的 格式 是 否 正确 。preg_match() 对 正则 表达 式 进 行 匹配 。 返回 匹配 的 
次 数 ，0 次 (没有 匹配 ) 或 者 1 次 ， 因 为 preg_match() 函 数 在 第 一 次 匹配 之 后 将 停止 搜索 ， 出 错 返回 false。 

目 $Submit2 一 "修改 ": 判断 表单 提交 的 值 是 否 等 于 “修改 "， 如 果 等 于 “修改 ” 则 执行 修改 的 操作 。 

四 S$Submit4--" 删 除 ": 判断 表单 提交 的 值 是 否 等 于 “删除 "， 如 果 等 于 “删除 ” 则 执行 删除 语句 。 


5.10 开发 技巧 与 难点 分 析 


5.10.1 应 用 存储 过 程 实现 管理 员 登 录 


在 物流 配送 信息 网 系统 中 ， 管 理 员 登 录 是 通过 存储 过 程 来 实现 的 。 下 面 就 来 讲解 一 下 如 何 创 建 存 
储 过 程 和 应 用 存储 过 程 。 

MySQL 5.0 以 后 的 版 本 中 开始 支持 存储 过 程 ， 存 储 过程 具 有 一 致 性 、 高 效 性 、 安 全 性 和 体系 结构 
等 特点 ， 其 将 在 以 后 的 项 目 中 得 到 广泛 的 应 用 。 

一 个 完整 的 存储 过 程 包括 名 称 、 参 数列 表 ， 以 及 很 多 SQL 语句 集 。 下 面 以 一 个 存储 过 程 的 定义 过 
程 为 例 ， 对 其 结构 进行 详细 的 讲解 。 


@ create procedure procedure_name (in parameter integer) 
@ begin 

declare variable varchar(20); 

if parameter=1 then 

set variable='Apache'; 

else 

set variable='MySQL '; 

end if; 

insert into tb_table (tb_name) values (variable); 

© end; 


Ah 代码 贴 十 
@ create procedure: 创建 存储 过 程 ， 以 关键 字 create procedure 开始 ， 后 面 紧 跟 存储 过 程 的 名 称 和 参数 。 
@ begin: MySQL 存储 过 程 的 语句 块 以 begin 开始 。 
@ end: MySQL 存储 过 程 的 语句 块 以 end 结束 。 


MySQL 中 存储 过 程 的 建立 以 关键 字 create procedure 开始 ， 后 面 紧 跟 存储 过 程 的 名 称 和 参数 。 
MySQL 的 存储 过 程 名 称 不 区 分 大 小 写 ,例如 PROCEDURES() 和 procedures() 代 表 同 一 存储 过 程 名 。 存 
储 过 程 名 不 能 与 MySQL 数据 库 中 的 内 建 函 数 重 名 。 

存储 过 程 的 参数 一 般 由 3 部 分 组 成 。 第 一 部 分 可 以 是 in、out 或 inout。in 表示 向 存储 过 程 中 传 入 
参数 ，out 表示 向 外 传 出 参数 ，inout 表示 定义 的 参数 可 传 入 存储 过 程 ， 并 可 以 被 存储 过 程 修改 后 传 出 
存储 过 程 。 存 储 过 程 默认 为 传 入 参数 ， 所 以 参数 in 可 以 省 略 。 第 二 部 分 为 参数 名 。 第 三 部 分 为 参数 的 类 
型 ， 该 类 型 为 MySQL 数据 库 中 所 有 可 用 的 字段 类 型 ， 如 果 有 多 个 参数 ， 参 数 之 间 可 以 用 逗号 进行 分 割 。 


@ 
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MySQL 存储 过 程 的 语句 块 以 begin 开始， 以 end 结束 。 语 句 体 中 可 以 包含 变量 的 声明 、 控 制 语句 、 
SQL 查询 语句 等 。 由 于 存储 过 程 内 部 语句 要 以 分 号 结束 ， 所 以 在 定义 存储 过 程 前 应 将 语句 结束 标志 “;” 
更 改 为 其 他 字符 ,并 且 该 字符 在 存储 过 程 中 出 现 的 几率 也 应 该 较 低 ,可 以 用 关键 字 delimiter 更 改 , 例如 : 


mysql>delimiter // 


a 
delimiter 关键 字 之 间 保 留 一 个 空格 ， 否 则 该 存储 过 程 将 会 出 现 错误 ， 不 能 成 功 创建 。 


存储 过 程 创建 之 后 ， 可 用 如 下 语句 进行 删除 ， 参 数 procedure_name 指 存 储 过 程 名 。 


drop procedure procedure_name 
MySQL 存储 过 程 是 在 命令 提示 符 下 创建 的 ， 其 操作 步骤 如 下 : 


(1) 打开 命令 提示 符 。 
(2) 进入 命令 提示 符 后 ， 首 先 登录 MySQL 数据 库 服务 器 ， 在 命令 提示 符 下 输入 如 下 命令 : 


mysql -u 用 户 名 -p 用 户 密码 


(3) 更 改 语句 结束 符号 ， 将 语句 结束 符 更 改 为 “/”， 代 码 如 下 : 


delimiter // 


(4) 创建 存储 过 程 前 应 首先 选择 某 个 数据 库 ， 代 码 如 下 : 


use 数据 库 名 


(5) 创建 存储 过 程 。 

(6) 通过 call 语句 调用 存储 过 程 。 

下 面 以 本 项 目 中 的 管理 员 登 录 模块 为 例 ， 对 存储 过 程 进 行 实例 化 的 讲解 。 首 先 创建 存储 过 程 ， 在 
命令 提示 符 中 输入 代码 ， 如 图 5.39 所 示 。 


图 539 创建 存储 过 程 


然后 在 index.php 文件 中 编写 PHP 脚本 语句 ， 调 用 存储 过 程 实现 用 户 登录 。 关 键 代 码 如 下 : 
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合 枉 15 ”代码 位 置 : 光盘 \TM\05\wlghindex.php 


<?php 
if($_POST[admin_user]!="™ || $_POST[admin_pass]!=™"X{ /获取 表单 提交 的 数据 
$conn=new mysqli("localhost","root","root',"db_wlgl"); // 连 接 数据 库 
$conn->query("set names gb2312"); // 指 定 字符 编码 格式 
$admin_user=$_POST[admin_user]; 1/ 获取 变量 值 
$admin_pass=md5($_POST[admin_pass]); // 获 取 变 量 值 
$sql=$conn->query("call admin_reg(".$admin_user.",".$admin_pass.")");// 调 用 存储 过 程 
$res=$sql->fetch_array(MYSQL_BOTH); // 输 出 结果 
if($res!=NULLY 
session_register("admin_user"); /定义 session 变量 
session_register("admin_pass"); 
echo "<script>alert( 用 户 注册 成 功 !");window.location.href='indexs.php';</script>"; 
Jelse{ 
echo "<script>alert(' 用 户 注册 失败 !");</script>"; 
} 
: 
ne 


5.10.2 ”应 用 正则 表达 式 验证 电话 号 码 


在 物流 配送 信息 网 中 ， 确 认输 入 的 电话 号 码 正 确 与 否 是 一 个 非常 重要 的 问题 ， 如 果 客 户 在 填写 订 
单 时 使 用 了 错误 的 电话 号 码 ， 那 么 就 会 带 来 很 多 不 必要 的 麻烦 ， 所 以 在 填写 电话 号 码 时 一 定 要 注意 电 
话 号 码 的 准确 性 。 为 了 更 好 地 确保 电话 号 码 输入 的 准确 性 ， 可 以 通过 程序 对 其 进行 控制 。 虽 然 程序 不 
能 判断 电话 号 码 输入 得 是 否 正确 ， 但 是 通过 程序 可 以 对 电话 号 码 的 格式 进行 判断 ， 从 而 避免 一 些 电话 
号 码 在 录入 的 过 程 中 出 现 多 写 或 者 漏 写 数字 的 错误 。 

验证 电话 号 码 格式 是 否 正确 可 以 通过 正则 表达 式 来 完成 。 通 过 正则 表达 式 可 以 有 效 地 判断 输入 电 
话 号 码 的 格式 是 否 正 确 。 

在 进行 判断 之 前 首先 要 对 电话 号 码 的 格式 进行 分 类 : 第 一 类 区 号 是 3 位 数字 ， 其 他 是 8 位 数字 ; 
第 2 类 区 号 是 4 位 数字 ， 其 他 是 7 位 数字 ; 第 3 类 区 号 是 4 位 数字 ， 其 他 是 8 位 数字 ; 第 4 类 是 手机 
电话 号 码 ， 由 11 位 数字 组 成 。 

然后 编写 一 个 判断 电话 号 码 格式 的 正则 表达 式 ， 其 代码 如 下 : 


人 (Gd(3}-)Gdft8DSIACd(-)Gdf7DSIAGd(4-)Gdf8)SIAGd(11)3/ 


正则 表达 式 的 功能 分 析 如 下 : 使 用 “^” 和 “$” 对 字符 串 进行 边界 限制 ， 对 区 号 从 字符 串 的 开始 
进行 匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 将 括号 “0” 中 的 内 容 作为 一 个 原子 使 用 ; 使 用 
“\d” 来 匹配 一 个 数字 ， 区 号 为 3 或 4 个 数字 ， 其 他 数字 为 7 或 8 个 ， 或 者 直接 判断 为 11 个 数字 ;使 
用 “{}” 来 对 前 字符 进行 重复 匹配 ;使 用 “|” 对 匹配 的 模式 进行 选择 ， 分 成 4 个 模式 。 

正则 表达 式 创建 完成 后 就 可 以 应 用 到 程序 中 对 电话 号 码 进行 判断 ， 要 完成 对 电话 号 码 的 判断 还 需 
要 应 用 preg_match() 函 数 。 

preg_match() 函 数 对 正则 表达 式 进行 匹配 。 返 回 匹 配 的 次 数 ，0 次 (没有 匹配 ) 或 者 1 次 ， 如 果 出 
错 则 返回 flse。preg_match() 函 数 的 参数 说 明 如 表 5.5 所 示 。 


第 5 章 物流 配送 信息 网 (Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


表 5.5 ”preg_match() 函 数 的 参数 说 明 


参数 说 了 明 
_pattem 必要 参数 ， 指 定 匹配 的 正则 表达 式 
subject 。 | 必要 参数 ， 指 定 要 搜索 的 字符 串 
可 选 参数 ， 如 果 提 供 了 subpattems， 则 其 会 被 搜索 的 结果 所 填充 。$ subpatterns [0] 将 包含 与 整个 模式 
subpattemns | 匹配 的 文本 ，$ subpatterns [1] 将 包含 与 第 一 个 捕获 的 括号 中 的 子 模式 所 匹配 的 文本 ， 以 此 类 推 
可 选 参数 , 如 果 设 定 标记 PREG_OFFSET_CAPTURE, 则 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 
flags 串 偏 移 量 。 注 意 这 改变 了 返回 数组 的 值 ， 使 其 中 的 每 个 单元 也 是 一 个 数组 ， 其 中 第 一 项 为 匹配 字符 串 ， 


第 二 项 为 其 偏 移 量 。 本 参数 自 PHP 4.3.0 开始 使 用 


最 后 将 正则 表达 式 和 preg_match() 函 数 应 用 到 实际 的 程序 中 ,实现 对 电话 号 码 格式 的 判断 。 这 里 以 
car_insert_ok.php 文件 中 的 代码 为 例 ， 关 键 代码 如 下 : 
倒 程 16 ”代码 位 置 ， 光盘 \TM\05\wlgl\car_insert_ok.php 
<?php 
if($Submit==trueX{ 1/ 判断 表单 提交 的 值 
六 oo 应 用 preg_match() 函 数 和 正则 表达 式 判 断 身 份 证 号 码 格式 是 否 正确 rr 让 
if(preg_match("Ad{17}N\d|X]I\d{15}/", $user_number,$counts)}{ 
eos 应 用 preg_match() 函 数 和 正则 表达 式 判 断 电话 号 码 格 式 是 否 正确 tr 4 
if(preg_match("/*(\d{3}-)(\d{8})$I^Qd{4}-)(\d{7})$I^(d{4}-)(\d{8})$I^(\d{11})$/", $user_tel,$countes)){ 
/ees 如 果 格 式 正确 ， 则 执行 下 面 的 插入 语句 ， 将 车 辆 信息 添加 到 数据 库 中 ******** */ 
$query=mysql_query("insert into tb_car(username,user_numbertel,address,car_numbercar_road, car_ 
content) values('$usemame','$user_number','$user_tel",'$address','$car_number','$car_road','$car_content')"); 
if($query==trueX{ 
echo "<script>alert(' 车 源 信息 添加 成 功 !);window.location.href='indexs.php?Imbs= 车 源 信息 管 理 '; 
</script>"; } 
}else{ 
echo "<script>alert(' 您 输入 的 电话 号 码 格式 不 正确 !);history.back()</script>"; } 
Jelse{ 


六 oressessesasssessessts 否 则 弹出 提示 信息 ， 返 回 到 表单 提交 页 seereseesresessesseesr 
echo "<script>alert(' 您 输入 的 身份 证 号 码 的 格式 不 正确 山 );historyback()</script>"; } 


5.11 报表 打印 技术 


报表 打印 技术 的 实现 方法 很 多 ， 可 以 通过 Web 打印 、Word 打印 、Excel 打印 和 CSS 样式 打印 等 。 
在 本 项 目 中 应 用 的 打印 方式 是 通过 Web 打印 ,并 且 是 打印 指定 框架 中 的 内 容 ,其 详细 的 讲解 可 参考 5.7.5 
节 。 下 面 再 介绍 一 种 通过 CSS 样式 来 打印 页 面 中 指定 内 容 的 方法 。 

通过 CSS 样式 打印 页 面 中 指定 内 容 主要 会 用 CSS 样式 的 media 类 型 。media 类 型 是 CSS 属性 媒体 
类 型 ， 用 于 直接 引入 媒体 的 属性 。 语 法 格式 如 下 : 


@media screen | print | projection | braille | aural | tv | handheld | all 


media 类 型 的 参数 说 明 如 表 5.6 所 示 。 


_ 国 
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表 5.6 media 类 型 的 参数 说 明 


参数 说 了 明 
screen 默认 值 ， 指 提交 到 计算 机 屏幕 
print 指 输出 到 打印 机 

Projection 指 提交 到 投影 机 

braille 提交 到 凸 字 触 觉 感知 设备 
aural 指 语音 电子 合成 器 

ty 电视 类 型 的 媒体 

handheld 指 手持 式 显示 设备 

all 用 于 所 有 媒体 


人 注意 应 用 C8s 科 风 厅 帮 本 和 网 是 机 十 表 从 克利 及 只 是 怀表 个 的 丰采 装 占 运行 克成 后 丰 
能 控制 表格 中 内 容 的 输出 。 所 以 在 应 用 CSS 样式 进行 指定 内 容 的 打印 时 ， 前 提 是 不 打印 的 内 容 必须 
是 以 表格 中 背景 图 像 的 形式 显示 的 ， 否 则 应 用 CSS 样式 进行 指定 内 容 打印 是 没有 效果 的 。 


这 里 以 项 目 中 的 客户 信息 为 例 ， 创 建 一 个 css_print.php 文件 ， 讲 解 通过 CSS 样式 打印 指定 内 容 的 
操作 过 程 。 首 先 编写 用 于 控制 指定 内 容 的 不 打印 CSS 样式 ， 程 序 代码 如 下 : 


<style> 

@media print{ 

div{display:none} 

.bgnoprint{ 
background:display:none; 

3 


.noprint{ 
display:none 

} 

3 

body{ 
margin-top: Opx; 


; 
</style> 


然后 设计 页 面 的 布局 ， 将 页 面 的 头 文件 和 尾 文 件 以 插入 背景 图 片 的 方式 进行 实现 ， 连 接 数据 库 ， 
应 用 while 循环 语句 输出 数据 库 中 的 客户 信息 ， 并 且 指定 页 面 的 头 文件 和 尾 文件 为 不 打印 内 容 。 


<body class="bgnoprint"> 
< 应 用 bgnoprint 样式 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一- > 
<table width="935" height="145" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<!-- 设 置 头 文件 ,以 背景 图 像 的 形式 进行 输出 ,并 且 应 用 bgnoprint 样式 , 控制 该 背景 图 像 为 不 打印 的 内 容 --> 
<td background="images/index_03.gif" class="bgnoprint">&nbsp;</td> 


</tr> 
</table> 
< 上 - 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 -连接 数据 库 ， 循 环 输出 数据 库 中 数据 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 > 
<?php 
include("conn/conn.php"); /| 连接 数据 库 


@ 
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$query=mysql_query("select * from tb_customer "); // 执 行 查询 语句 
while($myrow=mysql_fetch_array($query)X{ // 循 环 输出 数据 库 中 的 内 容 
?> 
<tr> 


<td height="26" align="center bgcolor="#FFFFFF"><?php echo $myrow[customer_user];?></td> 

<td align="center bgcolor="#FFFFFF"><?php echo $myrow[customer_tel];?></td> 

<td align="center" bgcolor="#FFFFFF"><?php echo $myrow[customer_address];?></td> 

<td align="center" bgcolor="#FFFFFF"><a href="delete_customer.php?delete=<?php echo 
$myrow[customer_id];?>"> 删 除 </a></td> 

</tr> 

<?php }?> 
</table> 


最 后 设置 一 个 div 标签 ， 建 立 超 链 接 ， 应 用 onClick 事件 调用 window.print() 方 法 和 printview() 函 数 
实现 页 面 的 打印 和 打印 预览 的 功能 。 


<!- 设 置 超 链接 ， 应 用 onClick 事件 ， 调 用 window.print() 方 法 执行 打印 ， 调 用 printview() 函 数 实现 打印 预览 的 功能 -> 
<tr> 

<td height="26" colspan="4" align="center" bgcolor="#FFFFFF"><div><a href="#" onClick= "window.print();"> 客 
户 详 单打 印 </a>&nbsp;<a href="# onClick="printview();"> 打 印 预览 </a></div></td> 
</tr> 


通过 CSS 样式 实现 页 面 中 指定 内 容 打印 的 运行 结果 如 图 5.40 和 图 5.41 所 示 。 


EHR 
记忆 丰 名 电 活 联系 地 直 | 
Em Ee [本 | Ea Cra 
图 5.40 客户 信息 打印 页 面 的 运行 结果 图 5.41 客户 信息 打印 预览 页 面 的 运行 结果 
本 


几 技巧 在 进行 Web 打印 时 ， 可 以 通过 以 下 操作 控制 是 否 打印 背景 颜色 和 图 像 。 在 下 窗口 中 ， 
选择 “工具 ”/“Intemet 选 项 ”命令 ， 在 弹出 的 “Intemet 选项 ”对 话 框 中 选择 “高 级 ”选项 卡 ， 在 
“设置 ”列表 中 设置 “打印 背景 颜色 和 图 像 ”前 面 的 复 选 框 是 否 选中 ， 如 果 选 中 ， 代 表 打 印 背 景 颜 
色 和 图 像 ， 否 则 不 打印 背景 颜色 和 图 像 。 


5.12 本 章 总 结 
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本 章 从 某 物流 配送 公司 的 实际 需求 角度 出 发 ， 开 发 一 个 完整 的 物流 配送 信息 管理 系统 ， 详 细 地 讲 
解 了 物流 配送 信息 网 的 开发 流程 ， 从 最 初 的 需求 分 析 、 可 行 性 分 析 ， 到 系统 的 设计 、 数 据 库 的 设计 ， 
其 中 重点 突出 车 源 信息 查询 模块 、 发 货 单 管理 模块 和 回执 单 验 收 管理 模块 的 设计 。 通 过 本 章 的 学 习 ， 
读者 不 但 可 以 了 解 物流 配送 系统 开发 的 整体 思路 ， 而 且 能 够 掌握 很 多 关键 的 技术 和 技巧 : 函数 的 灵活 
运用 、MySQL 存储 过 程 的 创建 和 应 用 、 报 表 打 印 技术 的 实现 和 应 用 正则 表达 式 验证 电话 号 码 等 。 


全 各 < 


GE 旱 


学 校 图 书馆 管理 系统 


(Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) 


随 着 网 络 技术 的 高 速 发 展 和 计算 机 应 用 的 善 及 ， 刊 用 计算 机 对 图 
书馆 的 日 常 工作 进行 管理 势 在 必 行 。 虽 然 目 前 很 多 大 型 的 图 书馆 已 经 
有 一 整套 比较 完善 的 管理 系统 ， 但 是 在 一 些 中 小 型 的 图 书馆 中 ， 大 部 
分 工作 仍 需 由 手工 完成 ， 工 作 起 来 效率 比较 低 ， 管 理 员 不 能 及 时 了 解 
图 书馆 内 各 类 图 书 的 借阅 情况 ,读者 需要 的 图 书 难 以 在 短 时 间 内 找到 ， 
不 便于 动态 及 时 地 调整 图 书 结构 。 为 了 更 好 地 适应 当前 读者 的 借阅 需 
求 ， 解 决 手工 管理 中 存在 的 许多 王 端 ， 越 来 越 多 的 中 小 型 图 书馆 正在 
逐步 向 计算 机 信息 化 管理 转变 。 本 章 通 过 开发 一 个 流行 的 图 书馆 管理 
系统 ， 为 读者 讲解 详细 的 项 目 开发 流程 。 通 过 阅读 本 章 ， 可 以 学 习 到 : 

M 图 书馆 管理 系统 天 发 的 基本 过 程 
系统 设计 的 方法 
如 何 分 析 并 设计 数据 库 、 数 据 表 
多 表 查 询 的 方法 
面向 对 象 的 编程 方法 
主要 功能 模块 的 实现 方法 
单元 测试 的 方法 
如 何 自 动 计算 图 书 娄 还 日 期 
内 联接 和 外 联接 语句 的 使 用 方法 


4 
至 


于 吾 于 于 于 于 至 


第 6 章 ”学校 图 书馆 管理 系统 ( Apache+PHP+phpMyAdmin+MySQL 5.0 实现 ) ”志和 毛 所 


6.1 开发 背景 


博 考 图 书馆 是 一 家 吉林 XX 师范 学 校 的 大 型 图 书馆 。 随 着 学 校 图 书馆 规模 的 不 断 壮 大 ， 经 营 的 图 
书 品 种 、 数 量 也 逐渐 增多 。 在 学 校 图 书馆 不 断 发 展 的 同时 ， 校 图 书馆 常年 采用 的 传统 的 人 工 方式 管理 


暴露 了 一 


些 问题 。 例 如 ， 碍 找 读者 借阅 的 某 一 本 图 书 的 具体 摆 放 位 置 ， 需 要 靠 人 工 记 忆 在 书 海中 兰若 


查找 ， 由 于 图 书 储存 量 大 ， 很 难 准确 定位 图 书 的 具体 位 置 ， 因 此 每 天 都 要 浪费 大 量 宝贵 的 时 间 资 源 。 
学 校 图 书馆 为 提高 工作 效率 ， 同 时 摆脱 图 书 管理 人 员 在 工作 中 出 现 的 种 种 次 端 ， 现 需要 委托 某 单位 开 
发 一 个 学 校 图 书馆 管理 系统 。 


6.2 需求 分 析 


通过 计算 机 对 图 书 进行 管理 ， 不 仅 为 图 书馆 的 管理 注入 了 新 的 生机 ， 而 且 在 运营 过 程 中 还 节省 了 
大 量 的 人 力 、 物 力 、 财 力 和 时 间 ; 不 仅 提 高 了 图 书馆 的 工作 效率 ， 还 为 图 书馆 在 读者 群 中 树立 了 一 个 
全 新 的 形象 ， 为 图 书馆 日 后 发 展 黄 定 了 一 个 良好 的 基础 。 通 过 对 一 些 大 型 图 书馆 的 实际 考察 、 分 析 ， 
并 结合 图 书馆 的 要 求 以 及 实际 的 市 场 调查 ， 要 求 本 系统 具有 以 下 功能 


因 因 办 办 办 办 办 办 囚 多 凶 


6.3.1 


网 站 设计 页 面 要 求 美观 大 方 、 个 性 化 ， 功 能 全 面 ， 操 作 简单 。 

要 求实 现 基础 信息 的 管理 平台 。 

要 求 对 所 有 读者 进行 管理 。 

要 求实 现 图 书 借阅 排行 、 了 解 当前 的 畅销 书 。 

商品 分 类 详尽 ， 可 按 不 同类 别 查看 图 书信 息 。 

提供 快速 的 图 书信 息 、 图 书 借阅 检索 功能 ， 保 证 数据 查询 的 灵活 性 。 

实现 图 书 借阅 、 图 书 续 借 、 图 书 归还 的 功能 。 

实现 综合 条 件 查询 ， 如 按 用 户 指定 条 件 查询 、 按 日 期 时 间 段 查询 、 综 合 条 件 查询 等 。 
要 求 图 书 借阅 、 续 借 、 归 还 时 记 下 每 一 笔记 录 的 操作 员 。 

实现 对 图 书 借阅 、 续 借 和 归还 过 程 的 全 程 数据 信息 跟踪 。 

提供 借阅 到 期 提醒 功能 ， 使 管理 者 可 以 及 时 了 解 到 已 经 到 达 归 还 日 期 的 图 书 借阅 信息 。 
提供 灵活 、 方 便 的 权限 设置 功能 ， 使 整个 系统 的 管理 分 工 明确 。 

具有 易 维 护 性 和 易 操 作 性 。 


6.3 系统 设计 


系统 目标 


根据 前 面 所 作 的 需求 分 析 及 用 户 的 需求 可 以 得 出 ， 学 校 图 书馆 管理 系统 实施 后 ， 应 达到 以 下 目标 : 


网 站 设计 页 面 要 求 美观 大 方 、 功 能 全 面 ， 操 作 简单 。 
网 站 整体 结构 和 操作 流程 合理 顺畅 ， 实 现 人 性 化 设计 。 
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规范 、 完 善 的 基础 信息 设置 。 

对 操作 员 设 置 不 同 的 操作 权限 ， 为 管理 员 提供 修改 权限 功能 。 
对 所 有 读者 进行 集中 管理 。 

对 图 书信 息 进行 集中 管理 。 

实现 图 书 借阅 排行 ， 以 便 了 解 当前 的 畅销 书 。 

提供 快速 的 图 书信 息 、 图 书 借阅 检索 功能 。 

rig 图 书 续 借 、 图 书 归还 功能 。 

现 综合 条 件 查询 ， 如 按 用 户 指定 条 件 查询 、 按 日 期 时 间 段 查询 、 综 合 条 件 查询 等 。 
实现 图 书 借 阅 、 续 代 、 归 还 时 记 下 每 一 笔记 录 的 操作 员 。 
支持 图 书 到 期 提醒 功能 。 

为 操作 员 提供 密码 修改 功能 。 
系统 运行 稳定 、 安 全 可 靠 。 


系统 功能 结构 


根据 学 校 图 书馆 管理 系统 的 特点 ， 可 以 将 其 分 为 系统 设置 、 读 者 管理 、 图 书 档案 管理 、 图 书 借 还 
系统 查询 等 5 个 部 分 ， 其 中 各 个 部 分 及 其 包括 的 具体 功能 模块 如 图 6.1 所 示 。 


学 校 图 书馆 管理 系统 
图 书 档案 管理 


因 因 办 办 办 办 办 办 办 办 多 
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图 | | 图 
书 || 书 
档 | | 借 
案 || 加 
查 | | 查 
询 | | 询 


图 6.1 学 校 图 书馆 管理 系统 功能 结构 图 
6.3.3 ”系统 流程 图 
学 校 图 书馆 管理 系统 的 流程 如 图 6.2 所 示 。 
6.3.4 系统 预览 


学 校 图 书馆 管理 系统 由 多 个 程序 页 面 组 成 ， 下 面 仅 列 出 几 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 
程序 。 

系统 登录 页 面 如 图 6.3 所 示 ， 该 页 面 用 于 实现 管理 员 登 录 。 系 统 首 页 如 图 6.4 所 示 ， 该 页 面 用 于 实 
现 显示 系统 导航 、 图 书 借阅 排行 和 版 权 信 息 等 功能 。 


@ 
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图 书 借阅 图 书 续 借 图 书 归还 


| 一 一 一 ”| 管理 读者 档案 信息 


1 


读者 档案 


下 


一 一 一 > 获取 读者 信息 


国信 宇 于 


图 6.3 系统 登录 (光盘 \TM\06\library\login.php) 图 6.4 系统 首页 (光盘 \TM\06\library\index.php) 


图 书 借阅 页 面 如 图 6.5 所 示 ， 该 页 面 用 于 实现 图 书 借阅 功能 。 图 书 借阅 查询 页 面 如 图 6.6 所 示 ， 该 
页 面 用 于 实现 按照 复合 条 件 查询 图 书 借阅 信息 。 


(BR 


art oi nt ee 
Hs pF sere 


图 6.5 图 书 借阅 (光盘 \TM\06\library\bookBorrow.php) 图 6.6 图书 借阅 查询 (光盘 \TM\06\library\borrowQuery.php) 
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6.3.5 ”开发 环境 


在 开发 学 校 图 书馆 管理 系统 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows Server 2003/Linux (推荐 )。 
服务 器 : Apache 2.2.8。 

PHP 软件 ，PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.10.3 。 
开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辩 率 : 最 佳 效果 为 1024x768 像素 。 


客户 端 


浏览 器 : IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效果 为 1024x768 像素 。 


6.3.6 ”文件 夹 组 织 结构 


在 编写 代码 之 前 , 可 以 把 系统 中 可 能 用 到 
的 文件 夹 先 创建 出 来 (例如 ， 创 建 一 个 名 为 
Images 的 文件 夹 , 用 于 保存 网 站 中 所 使 用 的 图 

片 )， 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 也 i 
可 以 规范 网 站 的 整体 架构 。 笔者 在 开发 学 校 图 ee 


因 办 办 办 办 办 办 轨 


I 


加 回 


用 于 存储 网 站 使 用 的 数据 库 连 接 文件 
用 于 存储 网 站 使 用 的 CSS 样 式 表 


书馆 管理 系统 时 ， 设 计 了 如 图 67 所 示 的 文件 a 
夹 组 织 结构 图 。 在 开发 时 ， 只 需要 将 所 创建 的 
文件 保存 在 相应 的 文件 夹 中 即 可 。 


6.4 数据 库 设计 


学 校 图 书馆 管理 系统 是 一 个 数据 库 开发 的 Web 网 站 。 下 面 对 学 校 图 书馆 使 用 的 数据 库 进行 分 析 和 
介绍 。 
6.4.1 数据 库 分 析 


由 于 本 系统 是 为 中 小 型 的 图 书馆 开发 的 程序 ， 需 要 充分 考虑 到 成 本 问题 及 使 用 需求 (如 跨 平 台 ) 
等 问题 ， 而 MySQL 是 世界 上 最 为 流行 的 开放 源码 的 数据 库 ， 是 完全 网 络 化 的 跨 平 台 的 关系 型 数据 库 
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系统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 
6.4.2 ”数据 库 概念 设计 
根据 以 上 各 节 对 系统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实体 分 别 为 图 书 档案 
实体 、 读 者 档案 实体 、 借 阅 档 案 实体 、 归 还 档案 实体 和 管理 员 实 体 。 下 面 将 介绍 儿 个 关键 实体 的 E-R 图 。 
1. 图 书 档案 实体 


图 书 档案 实体 包括 编号 、 条 形 码 、 书 名 、 类 型 、 作 者 、 译 者 、 出 版 社 、 价 格 、 页 码 、 书 架 、 录 入 
时 间 和 操作 员 等 属性 。 图 书 档案 实体 的 E-R 图 如 图 6.8 所 示 。 


2. 读者 档案 实体 


读者 档案 实体 包括 编号 、 姓 名 、 性 别 、 条 形 码 、 职 业 、 出 生日 期 、 有 效 证 件 、 证 件 号 码 、 电 话 、 
电子 邮件 、 登 记 日 期 、 操 作 员 、 类 型 和 备注 等 属性 。 读 者 档案 实体 的 E-R 图 如 图 6.9 所 示 。 


SA 
二 


图 6.8 图 书 档案 实体 的 E-R 图 图 6.9 读者 档案 实体 的 E-R 图 
3. 借阅 档案 实体 
借阅 档案 实体 包括 编号 、 读 者 编号 、 图 书 编号 、 借 书 时 间 、 应 还 时 间 、 操 作 员 和 是 否 归 还 等 属性 。 
借阅 档案 实体 的 E-R 图 如 图 6.10 所 示 。 
4. 归还 档案 实体 


归还 档案 实体 包括 编号 、 读 者 编号 、 图 书 编号 、 归 还 时 间 和 操作 员 等 属性 。 归 还 档案 实体 的 E-R 
图 如 图 6.11 所 示 。 


图 书 编号 
读者 编号 


图 6.10 ”借阅 档案 实体 的 E-R 图 图 6.11 “归还 档案 实体 的 E-R 图 
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6.4.3 创建 数据 库 及 数据 表 


结合 实际 情况 及 对 用 户 需求 的 分 析 , 学 校 图 书馆 管理 系统 db_library 数据 库 主 要 包含 10 个 数据 表 ， 
如 图 6.12 所 示 。 
结合 数据 表 的 创建 方法 ， 读 者 可 以 自行 创建 以 下 数据 表 。 


1. tb_bookinfo (图 书信 息 表 ) 
图 书信 息 表 主 要 用 于 存储 图 书 的 基础 信息 。 该 数据 表 的 结构 如 图 6.13 所 示 。 


图 服务 器 :localhost > 加 数据 库 : db_library * 国 表 : tb_bookinfo 
子 全 于 型 要 帮 all 轩 认 本 和 
bareode 。 varcharGol gb2312.chnese_el 再 图 条形码 
加 服务 器 : localhost ， 电 数据 库 : db_library peri D2 | OS 
表 类 型 整理 说 明 一 
paid ng) 是 NUL 必 书 关 开 
也 _bookcase 。 MyISAM 。 9b2312_chinese_cl 。 图 书 书架 信息 表 A CD TR mg pe 
也 _bookinfo 。 MyiSAM gb2312_chinese_cl 。 图 书信 息 表 ‘anstator = varcharGO gb2312_chnese_cl 是 NULL 图 湾 首 
也 _borrow MyISAM 。 9b2312_chinese_cl 。 图 书 僧 阅 信息 表 SBM -varcharQ0) gb2312_chnese cl 是。 AULL SBN 
由 bray 。 MyISAM gb2312_chinese_ci 。 图书 二 信息 表 price fcat@,D 旺 NULL 加 书 定价 
也 _manager 。 MyISAM 。 gb2312_chinese_cl 。 管理 员 信息 表 page mu 是 WUE 图 书页 码 
tb_parameter ”MyISAM ”gb2312_chinese_cl 。 参数 设置 信息 表 bookcase InK10) NULL 图书 书架 
也 _publishing 。 MyISAM gb2312_chinese_cl 。 出 版 社 信息 表 ss A 是 pt 二 
也 _purview MylSAM 。 gb2312_chinese_ci 。 权限 信息 表 rag BR ND 呈现 
tb_reader MyISAM ”gb2312_chinese_cl 。 读者 什 息 表 ool tyintc) 是 Mu 是 否 一 除 
队 _readertype MyISAM ”gb2312_chinese_cl 。 读者 类 型 信息 表 四 ny) 是 。 NULL auo_ncement 自动 号 
图 6.12 ”学校 图 书馆 管理 系统 数据 表 图 6.13 ”图 书信 息 表 结构 
2. tb_borrow (图 书 借阅 信息 表 ) 
ae 和 二 古寺 起 Ak 全 、 1 , 
图 书 借阅 信息 表 主 要 用 于 存储 图 书 的 借阅 信息 。 该 数据 表 的 结构 如 图 6.14 所 示 。 
3. tb_reader (读者 信息 表 ) 
读者 信息 表 主 要 用 于 存储 读者 的 基础 信息 。 该 数据 表 的 结构 如 图 6.15 所 示 。 
加 服务 器 :localhest ， 局 数据 库 : db_library ， 国 胡 :tb_reader 
#& Ra 天香 Nul 黑人 往外 说 明 
刘 nt(0) 理 auto_inerement ”自动 过 
ame varchar(20) 。 sb2312_chnese_ol 是 NULL 读者 姓名 
Sex Varthar() 。 cb2312_chnese_d 是 NULL 性 别 
barcode varchar(30) ”gb2312_chinese_d 是 MULL 读者 条 形 凤 
Vocation varthar(50) ”ob2312_chinese_dl 是 NULL 读者 职业 
inday cate 是 ”MULL 出 符 央 
中 服务 器 :localhost > 曲 数据 库 ; db_library ， 四 表 :tb_borrow paperlype 。 varthar(10) ”gb2312_chnese gd 是 WULL 读者 证 件 
Ea a E23] A 说 明 paperNO 。 varharGo) gb2312_chinese_a 是 NULL 正 作 号 码 
A tal varchar(20) ”gb2312_chinese_d 是 NULL 读者 电话 
Teadend mm 是 AL EE ee 
-一 一 一 | 一 email varchar(100) eb2312_chinese_d 是 WULL al 让 
borrowTime dale 是 MULL 图 书 们 同和 间 Me A 办 卡 时 间 
[eg NE BFE operator 。 vartharG0) gb2312_chinese_d 是 NULL 深 作 员 
operator rarcharGO gb2312_chinese_di 是 NULL 摊 作 盏 Temark mediumiex gb2312_chinese_d 是 NULL 名 注 
hack in) 是 0 是 ET ‘peid mt) 是 WULL 读者 类 型 
图 6.14 图 书 借阅 信息 表 结构 图 6.15 读者 信息 表 结 构 


/ 
说 明 限于 篇 幅 ， 笔 者 在 此 只 给 出 较 重要 的 数据 表 ， 其 他 数据 表 参 见 本 书 附 带 的 光盘 。 
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6.5 首页 设计 


6.5.1 首页 概述 


管理 员 通 过 系统 登录 模块 的 验证 后 ， 可 以 登录 到 图 书馆 管理 系统 的 首页 。 系 统 首页 主要 包括 导航 
栏 、 排 行 榜 和 版 权 信 息 3 部 分 。 其 中 ， 导 航 栏 中 的 功能 菜单 将 根据 登录 管理 员 的 权限 进行 显示 。 例如， 
系统 管理 员 Tsoft 登录 后 ， 将 拥有 整个 系统 的 全 部 功能 ， 因 为 它 是 超级 管理 员 。 

本 案例 中 提供 的 系统 首页 如 图 6.16 所 示 。 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\06\library\index.php。 


(半生 计 图 义 馆 管 理 系统 | 导航 栏 
i 


二 们 机 写作 省 行 磅 


并 各 图 和 放权 5 Bi 宰 


9 了 区 大 主 ZAFN Pen 排行 榜 


CD ”7 阳 攻 生 和 估 开 和 完全 和 颖 ”各 打 诛 注 二。 TID 生 宙 “rp 


el ?本 村 开 类 了 宇 由。 A 和。 


图 6.16 “学 校 图 书馆 管理 系统 首页 
6.5.2 首页 技术 分 析 


学 校 图 书馆 管理 系统 是 一 个 功能 全 面 、 大 型 的 Web 网 站 ， 通 过 对 网 站 的 安全 性 考虑 ， 本 网 站 对 该 
系统 进行 权限 的 分 配 ， 只 有 管理 员 级 别 的 超级 用 户 可 以 对 普通 用 户 的 权限 进行 管理 和 设置 。 系 统 首 页 
主要 通过 判断 管理 员 的 权限 来 显示 该 用 户 所 操作 的 功能 模块 ， 关 键 代 码 如 下 : 

倒 程 01 ”代码 位 置 : 光盘 \TM\06\library\navigation.php 


<?php 
session_start(); /初始化 session 变量 
include("conn/conn.php"); // 连 接 数据 库 文件 


$query=mysql_query("select m.id,m.name,p.id,p.sysset,p.readerset,p.bookset,p.borrowback,p.sysquery from th_manager 
as m left join (select * from tb_purview ) as p on m.id=p.id where name='$_SESSION[admin_name]"); 
$info=mysql_fetch_array($query); /检索 用 户 权 限 
3 
<!-- 检 索 用 户 所 对 应 的 权限 ， 如 果 权 限 值 为 1， 则 说 明 该 功能 可 用 ， 并 输出 到 浏览 器 ， 否 则 不 显示 --> 
<td width="70%" align="right"> 
<a href="index.php" class="a1"> 首 页 </a> | 
<?php if($info[sysset]==1){ ?><a onmouseover=showmenu(event,sysmenu) onmouseout=delayhidemenu() 
style="CURSOR:hand” class="a1"> 系 统 设置 </a> | <?php } ?> 
<?php if($info[readersetlj==1){?><a onmouseover=showmenu(event,readermenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="a1"> 读 者 管理 </a> | <?php } ?> 
<?php if($info[booksetj==1){ ?><a href="book.php" class="a1"> 图 书 档案 管理 </a> | <?php }?> 

四 
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<?php if($info[borrowbackjF==1){?><a onmouseover=showmenu(event,borrowmenu) onmouseout=delayhidemenu() 
style="CURSOR:hand"class="a1" > 图 书 借 还 </a> j <?php }?> 

<?php if($info[sysquery]==1){ ?><a onmouseover=showmenu(event,querymenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="a1"> 系 统 查询 </a> | <?php } ?> 

<a href="pwd_Modify.php" class="a1"> 更 改口 令 </a> | 

<a href="safequit.php" class="a1"> 注 销 </a> 

</td> 


说明 在 权限 信息 表 tb_purview 中 ， 权 限 值 为 1， 代表 具备 该 模块 的 操作 权限 ; 权限 值 为 0， 代 
表 不 具备 该 模块 的 操作 权限 .。 


在 实现 系统 导航 菜单 时 ， 引 用 了 JavaScript 文件 menu.JS， 该 文件 中 包含 全 部 实现 半 透 明 背 景 菜单 


的 JavaScript 代码 。 


Ls 技巧 将 页 面 中 所 涉及 的 JavaScript 代 码 保存 在 一 个 单独 的 JS 文件 中 , 然后 通过 <script></script> 
将 其 引用 到 需要 的 页 面 ， 可 以 规范 页 面 代码 。 在 系统 导航 页 面 引 用 menu.JS 文件 的 代码 如 下 : 


<script src="JS/menu.JS"></script> 


6.5.3 首页 的 实现 过 程 


系统 首页 的 内 容 显示 区 用 于 显示 图 书 的 排行 信息 ， 并 将 排行 结果 按 借阅 数量 降序 排列 。 该 页 的 关 


键 代码 如 下 : 


倒 程 02 代码 位 置 ， 光盘 \TM\06\library\index.php 


<?php 

include("conn/conn.php"); // 连 接 数 据 源 文件 
$sql=mysql_query("select * from (select bookid,count(bookid) as degree from tb_borrow group by bookid) as 
borr join (select b.*,c.name as bookcasename,p.pubname,ttypename from tb_bookinfo b left join tb_bookcase 
c on b.bookcase=c.id join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) 
as book on borr.bookid=book.id order by borr.degree desc limit 10"); 


$info=mysql_fetch_array($sql); // 检 索 图 书 借阅 信息 

$i=1; 

dof /应 用 do…while 循环 语句 显示 图 书信 息 
?> 

<tr> 


<td height="25" align="center"><?php echo $i;?></td> 

<td style="padding:5px;">&nbsp;<?php echo S$info[barcode];?></td> 
<td style="padding:5px;"><?php echo $info[bookname];?></td> 

<td style="padding:5px;"><?php echo $info[typename];?></td> 

<td align="center">&nbsp;<?php echo $info[bookcasename];?></td> 
<td align="center">&nbsp;<?php echo $info[pubname];?></td> 

<td align="center"><?php echo S$infofauthor];?></td> 

<td align="center"><?php echo S$info[price];?></td> 

<td align="center"><?php echo $info[degree];?></td> 


@ 
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</tr> 

<?php 
$i=$i+1; /变量 自 加 1 操作 
}while($info=mysql_fetch_array($sql)); /do…while 循环 语句 结束 


6.6 管理 员 模 块 设计 


6.6.1 管理 员 模 块 概述 


管理 员 模块 主要 包括 管理 员 登 录 、 查 看 管理 员 列 表 、 添 加 管理 员 信 息 、 管 理 员 权限 设置 、 管 理 员 
删除 和 更 改口 令 等 6 个 功能 。 管 理 员 模块 的 框架 如 图 6.17 所 示 。 


管理 员 列 表 


= 一 3 


添加 管理 员 权限 设置 删除 管理 员 [| mne | 


图 6.17 管理 员 模 块 的 框架 图 
6.6.2 ”管理 员 模 块 技 术 分 析 


在 管理 员 模 块 中 ， 涉 及 的 数据 表 是 tb_manager (管理 员 信 息 表 ) 和 tb_purview〔 权 限 表 )。 其 中 ， 
管理 员 信息 表 中 保存 的 是 管理 员 名 称 和 密码 等 信息 ， 权 限 表 中 保存 的 是 各 管理 员 的 权限 信息 ， 这 两 个 
表 通 过 各 自 的 id 字段 相关 联 。 通 过 这 两 个 表 可 以 获得 完整 的 管理 员 信息 。 


ri ER 
为 Tsoft, 密码 为 111， 拥有 所 有 权限 )， 即 在 MySQL 的 客户 端 命令 行 中 应 用 下 面 的 语句 分 别 向 管理 
员 信 息 表 tb_manager 和 权限 表 tb_purview 中 各 添加 一 条 数据 。 


# 添 加 管理 员 信 息 

insert into tb_manager (name,pwd) values(Tsoft ,111"); 
# 添 加 权限 信息 

insert into tb_purview values(1,1,1,1,1,1); 
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从 网 站 安全 的 角度 考虑 ， 仅 仅 有 上 面 介绍 的 系统 登录 页 面 并 不 能 有 效 地 保证 系统 的 安全 ， 一 旦 系 
统 首页 面 的 地 址 被 他 人 获得 ， 就 可 以 通过 在 地 址 栏 中 输入 系统 的 首页 面 地 址 而 直接 进入 到 系统 中 。 为 
了 便于 网 站 的 维护 ， 因 此 将 验证 用 户 是 否 登 录 的 代码 封装 在 独立 的 PHP 文件 中 ， 即 check_login.php 文 
件 。 验 证 用 户 是 否 登 录 的 具体 代码 如 下 : 

倒 程 03 ”代码 位 置 ， 光盘 \TM\06\library\check_login.php 


<?php 
session_start(); /初始 化 session 变量 
if($_SESSION[admin_name]==""){ /如 果 session 变量 为 空 , 则 说 明 用 户 未 登录 


echo "<script>alert(' 对 不 起 ， 请 通过 正确 的 途径 登录 博 考 图 书馆 管理 系统 !);window.location.href='|login. 
php';</script>"; 
} 
> 
当 系 统 调用 首页 时 ， 会 判断 session 变量 admin_name 是 否 存在 ， 如 果 不 存在 ， 将 页 面 重 定向 到 系 
统 登录 (login.php) 页 面 。 


6.6.3 ”系统 登录 的 实现 过 程 


国 。 系统 登录 使 用 的 数据 表 : tb_ manager 

系统 登录 是 进入 学 校 图 书馆 管理 系统 的 入 口 ， 主 要 用 
于 验证 管理 员 的 身份 。 运 行 本 系统 ， 首 先进 入 的 是 系统 登 
录 页 面 ， 在 该 页 面 中 ， 系 统管 理 员 可 以 通过 输入 正确 的 管 © 入 于 生得 险 
理 员 名 称 和 密码 登录 到 系统 首页 ， 当 用 户 没 有 输入 管理 员 
名 称 或 密码 时 ， 系 统 会 通过 JavaScript 进行 判断 ， 并 给 予 
信息 提示 。 系 统 登 录 页 面 的 运行 结果 如 图 6.18 所 示 。 

系统 登录 页 面 主要 用 于 收集 管理 员 的 输入 信息 及 通 图 6.18 系统 登录 页 面 的 运行 结果 
过 自 定义 的 JavaScript 函数 验证 输入 信息 是 否 为 空 。 该 页 
面 中 所 涉及 的 表单 元 素 如 表 6.1 所 示 。 


表 6.1 系统 登录 页 面 所 涉及 的 表单 元 素 


name text size="25" 管理 员 名 称 
pwd, password size="25" | 管理 员 密码 
submit submit value=" 确 定 " onclick="retum check(form1)" | “确定 ”按钮 


value=" 重 置 " 


value=" 关 闭 " onClick="window.close0:" 


“ 重 置 ”按钮 


reset 


编写 自 定义 的 JavaScript 函数 ， 用 于 判断 管理 员 名 称 和 密码 是 否 为 空 。 代 码 如 下 : 
倒 性 04 ”代码 位 置 ， 光盘 \TM\06\library\login.php 


<script language="javascript"> 
function check(formX{ // 自 定义 一 个 JavaScript 函数 check() 


他 
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if (form.name.value==""){ // 如 果 管 理 员 名 称 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 管理 员 名 称 ");form.name.focus();return false; 


} 
if (form.pwd.value==""){ // 如 果 管 理 员 密码 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 密码 ");form.pwd.focus();retum false; 


} 
》 


</script> 


提交 表单 到 数据 处 理 页 ， 页 面 中 为 了 防止 非法 用 户 进入 学 校 图 书馆 管理 系统 首页 ， 通 过 调用 类 的 
chkinput0 方 法 实现 判断 用 户 名 和 密码 是 否 正 确 。 如 果 为 合法 用 户 ， 则 可 以 登录 学 校 图 书馆 管理 系统 的 
首页 ; 否则 ， 弹 出 相应 的 错误 提示 。 关 键 代 码 如 下 : 

倒 程 05 ”代码 位 置 ， 光 盘 \TM\06\library\chklogin.php 


<?php 
session_start(); 


/初始化 session 变量 
$A_name=$_POSTIname]; /接收 表单 提交 的 用 户 名 
$A_pwd=$_POST[pwd]; // 接 收 表单 提交 的 密码 
@ classchkinput{ /定义 类 

var $name; 
var $pwd; 
function chkinput($x,$y){ /定义 一 个 方法 
Sthis->name=$x; // 将 管理 员 名 称 传 给 类 对 象 $this->name 
Sthis->pwd=$y; // 将 管理 员 密码 传 给 类 对 象 Sthis->pwd 
function checkinput(){ 
include("conn/conn.php"); // 连 接 数 据 库 文件 
$sql=mysql_query("select * fom tb_managerwhere name=".$this->name." and pwd=".$this->pwd."",$conn); 
$info=mysql_fetch_array($sql); // 检 索 管 理 员 名 称 和 密码 是 否 正确 


if($info==falseX{ // 如 果 管理 员 名 称 或 密码 不 正确 ， 则 弹出 相关 提示 信息 


echo "<script language='javascript>alert( 您 输入 的 管理 员 名 称 错误 ， 请 重新 输入 ! '); historyback(); 
</script>"; 


exit; 


} 
else{ // 如 果 管 理 员 名 称 或 密码 正确 ， 则 弹出 相关 提示 信息 
echo "<script>alert(' 管 理 员 登录 成 功 !");window.location='index.php';</script>"; 


$_SESSION[admin_name]=$info[name]; /将 管理 员 名 称 存 到 $_SESSION[admin_name] 变 量 中 


$_SESSION[pwd]=$info[pwd]; // 将 管理 员 密码 存 到 $_SESSION[pwd] 变 量 中 
} 

} 
} 

@ $obj=new chkinput(tim($name),trim($pwd)); // 创 建 对 象 
目  $obj->checkinput(); /调用 类 
和 

4 代码 贴 十 


@ class: 创建 一 个 PHP 类 时 ， 必 须 使 用 关键 字 class 进行 声明 ， 该 关键 字 后 紧 跟 类 的 名 称 ,之 后 用 大 括号 将 类 体 进 
行 封装 。 


@new: 在 PHP 中 应 用 new 关键 字 创 建 对 象 。 该 模块 创建 了 一 个 名 为 chkinput 的 验证 管理 员 类 。 
@ $obj->checkinput();: 通过 $obj 对 象 调用 checkinput 类 中 的 属性 和 方法 ， 即 管理 员 名 称 和 密码 。 
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6.6.4 查看 管理 员 的 实现 过 程 


国 查看 管理 员 使 用 的 数据 表 : tb_manager、tb_purview 

管理 员 登 录 后 ， 选 择 “ 系 统 设置 ”/“ 管 理 员 设 置 ”菜单 项 ， 进 入 到 查看 管理 员 列 表 页 面 。 在 该 页 
面 中 ， 将 以 表格 的 形式 显示 全 部 管理 员 及 其 权限 信息 ， 并 提供 添加 管理 员 信息 、 删 除 管理 员 信息 和 设 
置 管理 员 权限 的 超 链接 。 查 看 管理 员 列 表 页 面 的 运行 结果 如 图 6.19 所 示 。 


@ 全 二 周二 作 管 理 系统 


ot O07 vee nrtest oon 吉村 ev 大 学 图书 人 
二 让 请 全 RE 6 [上 叶 二 1086 人 有 全 风 村 


图 6.19 查看 管理 员 列表 页 面 的 运行 结果 
首先 使 用 左 外 联接 语句 (left join…on) 从 数据 表 tb_manager 和 tb_purview 中 查询 出 符合 条 件 的 数 
据 ， 然 后 将 查询 结果 应 用 do…while 循环 语句 输出 到 浏览 器 。 关 键 代 码 如 下 : 
倒 程 06 ”代码 位 置 ; 光盘 \TM\02\library\manager.php 


<?php 

include("conn/conn.php"); /| 连接 数据 库 文件 

© $sql=mysql_query("select m.id,m.name,p.sysset,p.readerset,p.bookset,p.borrowback,p.sysquery from 
tb_manager as m left join (select * from tb_purview) as p on m.id=p.id"); 


$info=mysql_fetch_array($sql); /检索 数据 信息 

do{ /应 用 do…while 循环 语句 输出 查询 结果 
?> 

nt 出 符合 查询 条 件 的 记录 ee 

<tr> 


<td style="padding:5px;"><?php echo S$info[name];?></td> 
四 <tdalign="center"><input name="checkbox" type="checkbox" class="noborder value="checkbox" 
disabled="disabled" <?php if($info[sysset]==1){echo ("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" 

disabled="disabled" <?php if($info[readerset]==1){echo("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info[bookset]==1){echo("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info[borrowback]==1){echo("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info[sysquery]==1){echo("checked");}?>></td> 

<td align="center><a href="#" onClick="window.open('manager_modify.php?id=<?php echo 
$info[id]; ?>',",'width=292,height=175'")"> 权 限 设 置 </a></td> 

<td align="center"><a href="manager_del.php?id=<?php echo $info[id];?>"> 删 除 </a></td> 

</tr> 
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<?php 


}while($info=mysql_fetch_array($sql)); 


?> 


/do…while 循环 语句 结束 
亏欠 代码 贴 二 
@ leftjoin…on: 应 用 左 外 联接 将 两 个 表 或 多 个 表 连 接 起 来 ， 返 回 部 分 或 全 部 匹配 行 ， 详 解 参 见 6.10.2 节 。 
@ <?php 这 $info[sysset] 一 1){echo ("checked");}?>: 如 果 系统 设置 字段 的 值 为 1， 则 复 选 框 处 于 选中 状态 。 
6.6.5 ”添加 管理 员 的 实现 过 程 


国 ” 添 加 管理 员 使 用 的 数据 表 : tb_manager 


图 书 要 大 


管理 员 登 录 后 ， 选 择 “ 系 统 设置 ”/“ 管 理 员 设 置 ”菜单 项 ， es 
进入 到 查看 管理 员 列表 页 面 ， 在 该 页 面 中 单 击 “ 添 加 管理 员 信息 ” 


加 | 2 


超 链接 , 打开 添加 管理 员 信 息 页 面 。 添 加 管理 员 信息 页 面 的 运行 结 
果 如 图 6.20 所 示 。 6.20 ”添加 管理 员 页 面 的 运行 结果 


于 和 


在 查看 管理 员 列表 页 面 ， 单 击 “ 添 加 管理 员 信 息 ” 超 链接 文字 的 HTML 代码 如 下 : 
倒 程 07 ”代码 位 置 ， 光盘 \TM\06\library\manager.php 


<a href="#" onClick="window.open('manager_add.php',",'width=292,height=175')"> 添 加 管理 员 信息 </a> 


添加 管理 员 页 面 主 要 用 于 收集 输入 的 管理 员 信 息 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 
否 合 法 。 该 页 面 中 所 涉及 的 表单 元 素 如 表 6.2 所 示 。 


表 6.2 添加 管理 员 页 面 所 涉及 的 表单 元 素 


名 称 元 素 类 型 含义 

forml form 表单 

name, text 管理 员 名 称 

pwd password 管理 员 密 码 

pwdl password 确认 密码 

submit submit value=" 保 存 " onClick="check(form1)" “保存 ”按钮 

Submit2 button value=" 关 闭 " onClick="window.close();" “关闭 ”按钮 
在 添加 管理 员 页 面 中 ， 输 入 合法 的 管理 员 名 称 及 密码 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 表 单 信息 到 数 


据 处 理 页 ， 将 添加 的 管理 员 信息 保存 到 数据 表 中 。 如 果 添 加 成 功 ， 弹 出 成 功 的 提示 信息 ; 否则 ， 弹 出 
错误 提示 。 代 码 如 下 : 


倒 程 08 ”代码 位 置 : 光盘 \TM\02\library\manager_ok.php 


<?php 

include("conn/conn.php"); /连接 数据 库 文件 

if($_POST[submit]!=""X // 如 果 单 击 了 “保存 ”按钮 ， 则 执行 下 面 的 操作 
$name=$_POST[name]; // 获 取 管 理 员 名 称 


_ 国 
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$pwd=$_POST[pwd]; // 获 取 管 理 员 密码 
$sql=mysql_query("insert into tb_manager (name,pwd) values('$name",'$pwd'")"); 
if($sql==true){ // 向 数据 表 中 添加 管理 员 信 息 成 功 ， 则 给 出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 添加 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; 
} 
else{ /向 数据 表 中 添加 管理 员 信息 失败 ， 则 给 出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 添加 失败 ! ');window.close();window.opener.location.reload(); 
</script>"; 
} 
} 
he 

2 


| 陋 技 


巧 在 添加 管理 员 处 理 页 中 应 用 “window.opener.location.reload();” 语 和 名 刷新 父 窗 口中 的 信息 。 


6.6.6 ”设置 管理 员 权 限 的 实现 过 程 


全 有 各 而， 阿 和 ] 


国 设置 管理 员 权限 使 用 的 数据 表 : tb_manager、tb_purview 三 邓 交 时 。 口 读者 和 理 

在 查看 管理 员 列表 页 面 单 击 指定 管理 员 后 面 的 “权限 设置 ” 超 
链接 ， 即 可 进入 到 “权限 设置 ”页 面 ， 设 置 该 管理 员 的 操作 权限 。 型 
权限 设置 页 面 的 运行 结果 如 图 6.21 所 示 。 SR 


权限 设置 页 面 中 所 涉及 的 表单 元 素 如 表 6.3 所 示 。 
表 6.3 ”权限 设置 页 面 所 涉及 的 表单 元 素 


checkbox, 
readerset checkbox, 2 
bookset checkbox, ?php iNSinfo[bookset] =—1) {echo("checked"):] 2> 
borrowback checkbox, <?php if($info[borrowback]=—=1) {echo("checked"):1?> 
Sysquery checkbox, <?php if($info[sysquery]==1){echo("checked"):}?> 
submit submit class="btn grey" 

Submit2 button value=" 关 闭 " onClick="window.close():;" 


在 查看 管理 员 列 表 页 面 中 ， 添 加 权限 设置 列 ， 并 在 该 列 中 添加 以 下 用 于 打开 权限 设置 页 面 的 超 链 
接 代码 。 
倒 程 09 ”代码 位 置 ， 光盘 \TM\02\library\manager.php 


<a href="#" onClick="window.open('manager_modify.php?id=<?php echo Sinfolid]; ?>",","width=292,height= 
175')"> 权 限 设置 </a> 


@ 
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从 上 面 的 URL 地 址 中 可 以 获取 设置 管理 员 权 限 页 所 涉及 的 这 号 ， 将 id 号 提交 给 处 理 页 manager_ 
modifyok.php， 修 改 id 号 所 对 应 的 管理 员 信 息 。 具 体 代 码 如 下 : 
倒 程 10 ”代码 位 置 ， 光盘 \TM\02\library\manager_modifyok.php 


<?php 
include("conn/conn.php"); /连接 数据 库 文件 
if($_POST[submitl!="")f // 如 果 提 交 表 单 ， 则 执行 以 下 操作 
$id=$_POST[id]; /获取 id 信息 
© $sysset=$_POST[sysset]==""?0:1; /应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 
$readerset=$_POST[readerset} ub // 应 用 三 目 运算 符 求 出 “读者 管理 ” 复 选 框 的 值 
$bookset=$_POST[bookset]==""?0:1; /应 用 三 目 运算 符 求 出 “图 书 管理 ” 复 选 框 的 值 
$borrowback=$_POST[borrowback]==""?0:1; /应 用 三 目 运算 符 求 出 “图 书 借 还 ” 复 选 框 的 值 
$sysquery=$_POST[sysquery]==""?0:1; /应 用 三 目 运算 符 求 出 “系统 查询 ” 复 选 框 的 值 
$query=mysql_query("select * from tb_purview where id=$id"); 

S$info=mysql_fetch_array($query); /检索 权限 信息 表 中 是 否 存在 该 管理 员 
if($info==falseX{ // 如 果 不 存 在 ， 向 权限 表 中 添加 管理 员 权限 信息 


四 ”mysql_query("insert into tb_purview(id,sysset,readerset,bookset,borrowback,sysquery) 
values($id,$sysset, $readerset,$bookset, $borrowback,$sysquery)"); 
; 
else{ // 否 则 ， 更 新 管理 员 的 权限 信息 
mysql_query("update tb_purview set 
Sysset=$sysset,readerset=$readerset,bookset=$bookset,borrowback=$borrowback,sysquery=$sysquery 
where id="$id"); 


} 

echo"<script language=javascript>alert(' 权 限 设置 修改 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; // 更 新 成 功 ， 弹 出 提示 信息 ， 并 更 新 父 窗口 

} 


?> 


< 代码 贴 十 
@$_POST[sysset] 一 ""?0:1;: 应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 ， 如 果 等 于 空 ， 值 为 0， 否 则 值 为 1。 
@ insert.…into; 向 权限 信息 表 中 添加 一 行 数据 信息 ，insert.…into 语句 只 适用 于 对 单行 数据 的 插入 。 


6.6.7 ”删除 管理 员 的 实现 过 程 


国 ”删除 管理 员 使 用 的 数据 表 : tb_manager、tb_purview 

在 查看 管理 员 列 表 页 面 中 ， 单 击 指定 管理 员 信息 后 面 的 “删除 ” 超 链 接 ， 该 管理 员 及 其 权限 信息 
将 被 删除 。 

在 查看 管理 员 列表 页 面 中 添加 以 下 用 于 删除 管理 员 信 息 的 超 链接 代码 。 

倒 程 11 代码 位 置 : 光盘 \TM\06\library\manager.php 

<a href="manager_del.php?id=<?php echo $info[id];?>"> 删 除 </a> 

从 上 面 的 URL 地 址 中 ， 可 以 获取 删除 管理 员 所 涉及 的 id 号, 将 id 号 提交 给 manager_del.php 处 理 
页 删除 id 号 所 对 应 的 管理 员 信息 。 有 具体 代码 如 下 : 

倒 程 12 ”代码 位 置 : 光盘 \TM\06\library\imanager_del.php 


<?php 
include("conn/conn.php"); // 连 接 数 据 库 文件 
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$id=$_GETI[id]; // 获 取 管理 员 的 id 号 

$sql=mysql_query("delete from tb_manager where id='$id"); /删除 管理 员 表 中 id 号 所 对 应 的 管理 员 信息 

$query=mysql_query("delete from tb_purview where id='$id"); // 删 除权 限 表 中 id 号 所 对 应 的 管理 员 权 限 

if($sql==true and $query==true }{ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 
echo "<script language=javascript>alert( 管 理 员 删除 成 功 ! '");history.back();</script>"; 


} 

else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 删除 失败 ! '");history.back();</script>"; 

} 

Ty 


.8 ”单元 测试 
在 开发 完 管理 员 模块 后 ， 需 要 对 该 模块 进行 单元 测试 。 当 管理 员 修改 操作 员 的 权限 时 ， 该 操作 员 


限 没有 改变 。 下 面 给 出 修改 权限 设置 处 理 页 的 源 代码 : 

<?php 

include("conn/conn.php"); // 连 接 数据 库 文件 
if($_POST[submit]l=") // 如 果 提 交 表 单 ， 则 执行 以 下 操作 
$id=$_POSTI[id]:; /获取 id 信息 
$sysset=$_POST[sysset]; 1/ 获取“ 系统 设置 ” 复 选 框 的 值 
$readerset=$_POST[readerset]; // 获 取 “ 读 者 管理 ” 复 选 框 的 值 
$bookset=$_POST[bookset]; // 获 取 “ 图 书 管理 ” 复 选 框 的 值 
$borrowback=$_POST[borrowback]; 1/ 获取“ 图书 借 还 ” 复 选 框 的 值 
$sysquery=$_POST[sysquery]; // 获 取 “ 系 统 查询 ” 复 选 框 的 值 
$query=mysql_query("select * from tb_purview where id=$id"); 

$info=mysql_fetch_array($query); /检索 权 限 信息 表 中 是 否 存在 该 管理 员 
if($info==falseX{ // 如 果 不 存在 ， 向 权限 表 中 添加 管理 员 权限 信息 


mysql_query("insert into tb_purview!(id,sysset,readerset,bookset,borrowback,sysquery) 
values($id,$sysset, $readerset,$bookset, $borrowback,$sysquery)"); 
} 
else{ 1/ 否则 ， 更 新 管理 员 的 权限 信息 
mysql_query("update tb_purview set 
Sysset=$sysset,readerset=$readerset,bookset=$bookset,borrowback=$borrowback,sysquery=$sysquery 
where id="$id"); 
3 
echo"<script language=javascript>alert( 权限 设置 修改 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; // 更 新 成 功 ， 弹 出 提示 信息 ， 并 更 新 父 窗口 
} 


?> 


为 了 找 出 错误 的 原因 ， 笔 者 应 用 echo() 语 句 对 $sysset 等 一 系列 传递 的 值 进行 输出 ， 输 出 的 结果 显 


示 $sysset 的 值 始 终 等 于 0， 这 充分 说 明 在 修改 复 选 框 时 ， 没 有 获取 到 选中 复 选 框 的 值 。 


解决 该 问题 的 方法 是 对 提交 的 复 选 框 的 状态 进行 判断 ， 如 果 复 选 框 处 于 未 选中 状态 ， 则 值 为 0; 如 


果 处 于 选中 状态 ， 则 值 为 1。 这 样 就 可 以 获取 到 复 选 框 选 中 状态 的 值 了 。 


处 理 复 选 框 的 状态 有 两 种 方法 : 一 种 是 应 用 让 条 件 语句 对 逐个 值 进行 判断 ， 判 断 复 选 框 的 当前 状 
并 赋予 其 值 ( 但 这 种 方法 过 于 繁琐 ); 另 一 种 是 应 用 三 目 运 算 符 ， 简 单 快捷 地 计算 复 选 框 的 值 ( 这 


@ 
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种 方法 可 以 简化 代码 ， 本 模块 采用 该 方法 获取 复 选 框 的 值 )。 
应 用 三 目 运算 符 获 取 复 选 框 的 值 的 代码 如 下 : 


$sysset=$_POST[sysset]==""?0:1; /应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 
S$readerset=$_POST[readerset]==""?0:1; // 应 用 三 目 运算 符 求 出 “读者 管理 ” 复 选 框 的 值 
$bookset=$_POST[bookset]==""?0:1; /应 用 三 目 运算 符 求 出 “图 书 管理 ” 复 选 框 的 值 
$borrowback=$_POST[borrowback]==""?0:1; /应 用 三 目 运算 符 求 出 “图 书 借 还 ” 复 选 框 的 值 
$sysquery=$_POST[sysquery]==""?0:1; // 应 用 三 目 运 算 符 求 出 “系统 查询 ” 复 选 框 的 值 


完整 的 权限 设置 处 理 页 的 代码 参见 例 程 10。 


6.7 图 书 档案 管理 模块 设计 


6.7.1 图 书 档案 管理 模块 概述 


图 书 档案 管理 模块 主要 包括 查看 图 书 列表 、 添 加 图 书信 息 、 修 改 图 书信 息 、 删 除 图 书信 息 和 查看 
图 书 详细 信息 等 5 个 功能 。 图 书 档案 模块 的 框架 如 图 6.22 所 示 。 


图 书信 息 列 表 
查看 图 书 列表 添加 图 书信 息 修改 图 书信 息 删除 图 书信 息 查看 图 书 详细 信息 


图 622 图 书 档案 模块 的 框架 图 
6.7.2 图书 档 案 管理 模块 技术 分 析 


在 图 书 档案 管理 模块 中 ,涉及 的 数据 表 是 tb_bookinfo 图 书信 息 表 )、tb_bookcase( 书 架设 置 表 )、 
tb_booktype( 图 书 类 型 表 ) 和 tb_publishing( 出 版 社 信息 表 )， 这 4 个 数据 表 间 通过 相应 的 字段 进行 关 
联 ， 如 图 6.23 所 示 。 通 过 以 上 4 个 表 可 以 获得 完整 的 图 书 档案 信息 。 


hbookinko 
ateger C1) 


pubnane varchar (30) 


hbookease 
i inteeer aasiened GE 
[ame varchar GO) 


图 623 图 书 档案 管理 模块 各 表 间 关系 图 
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6.7.3 ”查看 图 书信 息 列 表 的 实现 过 程 


国 坦 看 图 书信 息 列表 使 用 的 数据 表 : tb_bookinfo、tb_bookcase、tb_booktype、tb_publishing 

管理 员 登 录 后 ， 选 择 “ 图 书 管理 ”/“ 图 书 档案 管理 ”菜单 项 ， 进 入 到 查看 图 书 列表 页 面 ， 在 该 页 
面 中 将 显示 全 部 图 书信 息 列表 ， 同 时 提供 添加 图 书信 息 、 删 除 图 书信 息 、 修 改 图 书信 息 的 超 链 接 。 查 
看 图 书信 息 列 表 页 面 的 运行 结果 如 图 6.24 所 示 。 
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本 节 衣 便 Hz 5 环 以 上 可 10S9vT96 罚 人 有 于 天 和 


图 6.24 查看 图 书信 息 列表 的 运行 结果 
打开 功能 导航 navigation.php 文件 ， 设 置 “ 图 书 档案 管理 ”菜单 项 的 超 链接 的 代码 如 下 : 
倒 程 13 ”代码 位 置 ， 光盘 \TM\02\library\navigation.php 
<a href="book.php" class="a1"> 图 书 档案 管理 </a> 


首先 应 用 join…on 内 联接 语句 将 tb_bookinfo、tb_bookcase、tb_booktype 和 tb_publishing 4 个 数据 
表 连 接 起 来 检索 指定 条 件 的 图 书信 息 , 然后 应 用 do…while 循环 语句 输出 查询 结果 到 浏览 器 。 查 看 图 书 
信息 页 面 的 代码 如 下 : 

倒 程 14 ”代码 位 置 ， 光盘 \TM\06\library\book.php 

<?php 

include("conn/conn.php"); /连接 数据 库 文件 

$query=mysql_query("select book.barcode,book.id as bookid,book.bookname,bt.typename,pb.pubname, bc. 


name from tb_bookinfo book join tb_booktype bt on book.typeid=bt.id join tb_publishing pb on book.ISBN= 
pb.ISBN join tb_bookcase bc on book.bookcase=bc.id"); 


$result=mysql_fetch_array($query); /应 用 外 联接 检索 图 书信 息 
J 
// 省 略图 书信 息 标题 HTML 标记 部 分 
<?php 
dof /应 用 do…while 循环 语句 输出 查询 结果 
?> 
<tr> 


<td style="padding:5px;">&nbsp;<?php echo $result[barcode];?></td> 
<td style="padding:5px;"><a href="book_look.php?id=<?php echo $resultfbookid];?>"><?php echo 
$result[lbookname];?></a></td> 
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<td style="padding:5px;">&nbsp;<?php echo $result[typename];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[pubname];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $resultIname];?></td> 
<td align="center"><a href="book_Modify.php?id=<?php echo $result[bookid];?>"> 修 改 </a></td> 
<td align="center"><a href="book_del.php?id=<?php echo $result[bookid];?>"> 删 除 </a></td> 
</tr> 
<? 
}while($result=mysql_fetch_array($query)); /do…while 循环 语句 结束 
学 > 


人 注意 x 于 join.…on 内 联接 语句 的 使 用 方法 参见 610.1 节 。 


6.7.4 添加 图 书信 息 的 实现 过 程 


国 ” 添 加 图 书信 息 使 用 的 数据 表 : tb_bookinfo、tb_bookcase、tb_booktype、tb_publishing 

管理 员 登 录 系 统 后 ， 在 导航 栏 中 单 击 “ 图 书 档案 管理 ” 超 链接 ， 进 入 到 查看 图 书 列表 页 面 。 在 该 
页 面 中 单 击 “ 添 加 图 书信 息 ” 超 链接 ， 进 入 到 添加 图 书信 息 页 面 。 添 加 图 书信 息 页 面 的 运行 结果 如 
图 6.25 所 示 。 


@ 他 专 周 专人 管理 系统 
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图 6.25 添加 图 书信 息 页 面 的 运行 结果 
在 查看 图 书 列表 页 面 中 ， 设 置 “ 添 加 图 书信 息 ” 超 链接 的 代码 如 下 : 


倒 程 15 ”代码 位 置 ， 光盘 \TM\06\library\book.php 
<a href="book_add.php"> 添 加 图 书信 息 </a> 


添加 图 书信 息 页 面 主要 用 于 收集 输入 的 图 书信 息 以 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 
是 否 合 法 。 该 页 面 中 所 涉及 的 重要 表单 元 素 如 表 6.4 所 示 。 


表 6.4 添加 图 书信 息 页 面 所 涉及 的 重要 表单 元 素 


重要 属性 


forml method="post" action="book_ok.php" | 表单 
<?php include("Conn/conn.php"); 书 类 型 


typeld 


"select * from tb booktype"); 
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续 表 
名 称 元 素 类 型 重要 属性 含 义 
Sinfo=mysql_fetch_array($sq]); 
dof 
typeld select > 图 书 类 型 
<option value="<?php echo $info[id];?>"> 
<?php echo Sinfo[typename]:;?></option> 
<?php !while($info=mysql fetch array($sql)):?> 
<?php 
$sql2=mysql_query("select * from tb_publishing"); 
Sinfo2=mysql]_fetch_array($sql2); 
isbn 人 > 出 版 社 
<option value="<?php echo $info2[ISBN]:?>"> 
<?php echo $info2[pubname];?></option> 
<?php }while($info2=mysql_fetch_array($sql2)):?> 
<?php 
$sql3=mysql_query("select * from tb_bookcase"); 
Sinfo3=mysql_fetch_array($sq13); 
bookcaseid select 小 人 书架 名 称 
<option value="<?php echo $info3[id];?>"> 
<?php echo $info3[name];?></option> 
<?php } while(S$info3=mysql fetch array($sgl3)):;?> 
Operator | hidden | value="<?php echo $info3[name]:?>" 操作 员 
Submit | submit | onClick="return check(form1)" “保存 ”按钮 
Submit2 | bution | onClick="history.back():" “返回 ”按钮 
由 于 添加 图 书信 息 的 方法 同 添加 管理 员 信 息 的 方法 类 似 ， 所 以 此 处 只 给 出 向 图 书信 息 表 中 插入 数 


据 的 SQL 语句 ， 详 细 代 码 参见 光盘 。 向 图 书信 息 表 中 插入 数据 的 SQL 语句 如 下 : 
倒 程 16 ”代码 位 置 ， 光盘 \TM\06\library\book_ok.php 


mysql_query("insert into 
tb_bookinfo(barcode,bookName,typeid,author,translator,ISBN, price, page,bookcase,inTime,operator )values($b 
arcode','$bookName','$typeid','$author,'$translator','$isbn','$price','$page','$bookcaseid','$inTime','$operator’)"); 


6.7.5 修改 图 书信 息 的 实现 过 程 


国 修改 图 书信 息 使 用 的 数据 表 : tb_bookinfo、tb_bookcase、tb_booktype、tb_publishing 

管理 员 登 录 系 统 后 ， 在 导航 栏 中 单 击 “ 图 书 档案 管理 ” 超 链 接 ， 进 入 到 查看 图 书 列 表 页 面 。 单 击 
想 要 修改 的 图 书信 息 后 面 的 “修改 ” 超 链 接 ， 进 入 到 “修改 图 书信 息 ” 页 面 。 修 改 图 书信 息 页 面 的 运 
行 结 果 如 图 6.26 所 示 。 
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在 图 书信 息 列表 页 面 中 ， 添 加 “修改 ” 超 链接 的 代码 如 下 : 
全 性 17 ”代码 位 置 ， 光盘 \TM\06\library\book.php 


<a href="book_Modify.php?id=<?php echo $result[bookid];?>"> 修 改 </a> 


在 修改 图 书信 息 页 面 中 修改 图 书信 息 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 表 单 信息 到 数据 处 理 页 
book_Modify_ok.php， 应 用 UPDATE 语句 将 修改 的 图 书信 息 保存 到 数据 表 tb_bookinfo 中 ， 并 弹出 “图 


书信 息 修改 成 功 !” 提 示 信 息 ， 将 页 面 重 定向 到 修改 图 书信 息 页 。 数 据 处 理 页 的 代码 如 下 : 
倒 程 18 ”代码 位 置 ， 光盘 \TM\06\library\book_Modify_ok.php 


<?php 

session_start(); /| 初始 化 session 变量 
include("conn/conn.php"); // 连 接 数 据 库 文件 
$bid=$_POST[bid]; /获取 图 书 id 号 
$operator=$_SESSION[admin_name]; // 获 取 管 理 员 名 称 
$barcode=$_POST[barcode]; // 获 取 图 书 条 形 码 
$bookName=$_POST[bookName]; /获取 图 书 名 称 
$typeid=$_POST[typeld]; // 获 取 图 书 类 型 id 号 
$author=$_POST[author]; /获取 图 书 作 者 
S$translator=$_POSTI[translator]; /获取 图 书 译 者 
Sisbn=$_POST[isbn]; // 获 取出 版 社 ISBN 


$price=$_POSTIprice]; 
$page=$_POST[Ipage]; 
$bookcase=$_POST[bookcaseid]; // 获 取 图 书 书架 id 号 

S$inTime=date("Y-m-d"); 1/ 设置 图 书 更 新 日 期 为 当前 日 期 


// 获 取 图 书 单价 


$query=mysql_query("update tb_bookinfo set barcode='$barcode', bookName='$bookName' , typeid='$typeid', 
author='$author', translator='$translator, ISBN='$isbn', price='$price' , page='$page' , bookcase='"$bookcaseid', 


inTime='$inTime', operator='$operator where id=$bid"); /更 新 数据 表 
echo "<script language='javascript>alert( 图 书信 息 修改 成 功 !");history.back();</script>"; 


?> 


6.7.6 ”删除 图 书信 息 的 实现 过 程 


国 ”删除 图 书信 息 使 用 的 数据 表 : tb_bookinfo 
在 查看 图 书 列表 页 面 中 ， 设 置 “ 删 除 ” 超 链接 的 代码 如 下 : 
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倒 程 19 ”代码 位 置 ， 光盘 \TM\06\library\book.php 
<a href="book_del.php?id=<?php echo $result[bookid];?>"> 删 除 </a> 


单 击 想 要 删除 的 图 书信 息 后 面 的 “删除 ” 超 链 接 ， 提 交 表单 信息 到 数据 处 理 页 book_del.php， 应 用 
DELETE 语句 将 指定 的 图 书信 息 从 数据 表 tb_bookinfo 中 删除 ， 如 果 删 除 操作 执行 成 功 ， 则 弹出 “图 书 
信息 删除 成 功 !” 提 示 信息 ， 并 将 页 面 重 定向 到 图 书信 息 列表 页 面 。 数 据 处 理 页 的 代码 如 下 : 

倒 程 20 ”代码 位 置 : 光盘 \TM\06\library\book_del.php 
<?php 

include("conn/conn.php"); 


// 连 接 数据 库 文件 
S$info_del=mysql_query("delete from tb_bookinfo where id=$_GET[id]"); /删除 指定 的 图 书信 息 
if(Ginfo_del}{ 


// 如 果 信 息 删 除 成 功 , 则 弹出 提示 
echo "<script language='javascript'>alert(' 图 书信 息 删 除 成 功 !");history.back();</script> "; 
让 


?> 


6.8 图 书 借 还 模块 设计 
6.8.1 图 书 借 还 模块 概述 


图 书 借 还 模块 主要 包括 图 书 借阅 、 图 书 续 借 、 图 书 归 还 、 图 书 档案 查询 、 图 书 借阅 查询 、 借 阅 到 
期 提醒 等 6 个 功能 。 在 图 书 借阅 模块 中 的 用 户 只 有 一 种 身份 ， 那 就 是 操作 员 ， 通 过 该 身份 可 以 进行 图 
书 借 还 等 相关 操作 。 图 书 借 还 模块 的 用 例 图 如 图 6.27 所 示 。 
< 
CFRP + 人 < 
< ca 
图 6.27 图 书 借 还 模块 的 用 例 图 


6.8.2 图 书 借 还 模块 技术 分 析 


在 图 书 借 还 模块 中 涉及 的 数据 表 是 tb_borrow 〈 图 书 借阅 信息 表 )、tb_bookinfo (图 书信 息 表 ) 和 
tb_reader (读者 信息 表 )， 这 3 个 数据 表 间 通过 相应 的 字段 进行 关联 ， 如 图 6.28 所 示 。 
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int 


tyint 0) 


图 6.28 图 书 借 还 管理 模块 各 表 间 关系 图 
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a 


6.8.3 图 书 借阅 的 实现 过 程 


国 。 图书 借 阅 使 用 的 数据 表 : tb_borrow、tb_bookinfo、tb_reader 

管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 借阅 ”菜单 项 ， 进 入 到 图 书 借阅 页 面 ， 在 该 页 面 的 “ 读 
者 条 形 码 ”文本 框 中 输入 读者 的 条 形 码 (如 123456789) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 
读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 将 其 显示 在 页 面 中 ， 此 时 输 
入 图 书 的 条 形 码 或 图 书 名 称 后 ， 单 击 “ 确 定 ” 按 钮 ， 借 阅 指 定 的 图 书 ， 运 行 结果 如 图 6.29 所 示 。 
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6.29 图 书 借阅 页 面 的 运行 结果 


4 
说 明 当 读 者 借阅 图 书 完毕 后 ， 操 作 员 通过 单 击 “ 完 成 借阅 ”按钮 ， 将 重新 载 入 图 书 借阅 页 面 ， 
当前 页 处 于 空 信息 状态 ， 从 而 方便 操作 员 对 下 一 个 读者 进行 借阅 图 书 操作 。 


图 书 借阅 页 面 总 体 上 可 以 分 为 两 个 部 分 : 一 部 分 用 于 查询 并 显示 读者 信息 另 一 部 分 用 于 显示 读 
者 的 借阅 信息 和 添加 读者 借阅 信息 。 图 书 借阅 
页 面 在 Dreamweaver 中 的 设计 效果 如 图 6.30 
在 进行 图 书 借阅 时 ， 系 统 要 求 每 个 读者 只 
能 同时 借阅 一 定数 量 的 图 书 , 并 且 该 数量 由 读 
者 类 型 表 tb_readertype 中 的 可 借 数 量 number 
决定 ， 所 以 笔者 编写 了 自 定义 的 checkbook0 图 6.30 图 书 借阅 页 面 的 设计 效果 
函数 ,用 于 判断 当前 选择 的 读者 是 否 还 可 以 借 
阅 新 的 图 书 ， 同 时 该 函数 还 具有 判断 输入 读者 条 形 码 或 图 书 名 称 文本 框 是 否 为 空 的 功能 。 代 码 如 下 : 
倒 程 21 ”代码 位 置 : 光盘 \TM\02\bookBorrow.php 


<script language="javascript"> 
function checkbook(formX{ // 自 定义 一 个 JavaScript 函数 checkbook() 
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if(form.barcode.value=="){ /如果 读 者 条 形 码 为 空 
alert(" 请 输入 读者 条 形 码 ");form.barcode.focus();return; // 弹 出 提示 ， 焦 点 返回 到 条 形 码 文本 框 

长 

if(form_.inputkey.value==""){ /如果 图 书 查询 文本 框 的 值 为 空 
alert(" 请 输入 查询 关键 字 "");form.inputkey.focus();return; // 弹 出 提示 ， 焦 点 返回 到 图 书 查询 文本 框 

} 

if(form.number.value-form.borrowNumber.value<=0){ // 如 果 图 书 的 借阅 数量 超过 了 可 借 数 量 
alert(" 您 不 能 再 借阅 其 他 图 书 了 !"");return; // 弹 出 提示 信息 

} 

form.submit(); /提交 表单 
} 
</script> 


AM 


[> 技巧 在 JavaScript 中 比较 两 个 数值 型 文本 框 的 值 时 ， 不 使 用 运算 符 “ 一 ”， 而 是 将 这 两 个 值 相 
减 ， 再 判断 其 结果 。 


检索 读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 的 SQL 语句 如 下 : 
倒 性 22 ”代码 位 置 ， 光盘 \TM\06\library\bookborrow.php 


$sql=mysql_query("select r.*,t.name as typename,tnumber from tb_reader r left join tb_readerType t on 
r.typeid=t.id where rbarcode='$barcode"); 
$info=mysql_fetch_array($sql); /检索 读者 信息 


获取 读者 借阅 信息 的 SQL 语句 如 下 : 
倒 程 23 ”代码 位 置 ， 光盘 \TM\06\library\bookborrow.php 


$sql1=mysql_query("select r.*,borr.borrowTime,borr.backTime,book.bookname,book.price,pub.pubname,bc. name 
as bookcase from tb_borrow as borr join tb_bookinfo as book on book.id=borr.bookid join tb_publishing as pub 
on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id join tb_reader as r on borr.readerid= 
rid where borrreaderid='$readerid' and borr.ifback=0"); 

$info1=mysql_fetch_array($sql1); /检索 读者 的 借阅 信息 
$borrowNumber=mysql_num_rows($sql1); /获取 结果 集中 行 的 数目 


在 “图 书 条 形 码 ”/“ 图 书 名 称 ” 文 本 框 中 输入 图 书 条 形 码 或 图 书 名 称 后 ， 单 击 “ 确 定 ” 按 钮 ， 检 
索 图 书信 息 是 否 存 在 ， 如 果 不 存在 ， 则 向 图 书 借阅 信息 表 中 添加 该 读者 的 图 书 的 借阅 记录 ， 完 成 图 书 


借阅 操作 ;， 和 否则， 弹出 该 书 不 能 被 同一 读者 重复 借阅 的 提示 信息 。 图 书 借阅 的 具体 代码 如 下 : 

倒 程 24 ”代码 位 置 : 光盘 \TM\06\library\bookBorrow.php 
<?php 

if($_POSTIinputkey]!=""){ /| 如果“ 图 书 条 形 码 ”/“ 图 书 名 称 ” 文 本 框 不 为 空 
$f=$_POSTIN; // 获 取 用 户 选 择 的 条 件 值 
@ $inputkey=trim($_POST[inputkey]); // 获 取 用 户 输入 的 查询 关键 字 
$barcode=$_POST[barcode]; /获取 读者 的 条 形 码 
$readerid=$_POST[readerid]; // 获 取 读 者 id 号 
@ $borrowTime=date(Y-m-d"); /图书 的 借阅 时 间 为 系统 当前 时 间 


@ $backTime=date("Y-m-d",(time()+3600*24*30)); ”// 归 还 图 书 日 期 为 当前 期 日 期 +30 天 期 限 
$query=mysql_query("select * from tb_bookinfo where $f='$inputkey™); 
$result=mysql_fetch_array($query); // 检 索 图 书信 息 是 否 存在 


@ 
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if($result==false){ // 如 果 读 者 借阅 的 图 书 不 存在 ， 那 么 弹出 提示 信息 
echo "<script language='javascript>alert(' 该 图 书 不 存在 ! 
");window.location.href='bookBorrow.php?barcode=$barcode'; </script>"; 


} 
else{ // 检 索 该 读者 所 借阅 的 图 书 是 否 与 再 借 图 书 重复 
$query1=mysql_query("select r.*,borr.borrowTime,borr.backTime,book.bookname,book.price, pub.pubname, bc. 
name as bookcase from tb_borrow as borr join tb_reader as r on borrreaderid=rid join tb_bookinfo as book on 
book.id=borrbookid join tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book. 
bookcase=bc.id where borrbookid=$resultfid] and borrreaderid=$readerid and ifback=0"); 
$result1=mysql_fetch_array($query1); 
if($result1==trueX{ // 如 果 所 借 图 书 已 被 该 读者 借阅 ， 那 么 提示 不 能 重复 借阅 
echo "<script language=javascript>alert(' 该 图 书 已 经 借阅 ! ');window.location.href='bookBorrow.php? 
barcode=$barcode'</script>"; 
bs 
else{ // 否 则 ， 完 成 图 书 借阅 操作 ， 并 弹出 借阅 成 功 提示 信息 
$bookid=$resultfid]; // 将 读者 id 号 赋 给 一 变量 
mysql_query("insert into 
tb_borrow(readerid,bookid,borrowTime,backTime,operator,ifback)values('$readerid','$bookid','$borrowTime','$back 
Time','$_SESSION[admin_name]',0)"); // 向 借阅 信息 表 中 添加 一 条 借阅 信息 
echo "<script language='javascript>alert(' 图 书 借阅 操作 成 功 ! 
);window.location.href='bookBorrow.php?barcode=$barcode';</script>"; 
} 
是 


?> 


< 代 码 贴 十 

@ trim(): 删除 字符 串 中 首尾 的 空白 或 者 其 他 字符 ， 以 达到 精确 查询 。 

@ date(Y-m-d): 获取 系统 的 当前 日 期 为 图 书 借阅 的 日 期 ， 并 格式 化 日 期 格式 。 

@ date("Y-m-d",(time()+3600*24*30)): 归还 图 书 日 期 = 当前 期 日 期 130 天 期 限 . 当前 日 期 时 间 蕉 应 用 time() 函 数 获取 ， 
30 天 期 限 的 时 间 鹤 等 于 3600 秒 x24 小 时 x30 天， 并 通过 date() 函 数 格式 化 为 指定 日 期 格式 。 


6.8.4 图 书 续 借 的 实现 过 程 


国 。 图书 续 借 使 用 的 数据 表 : tb_borrow、tb_bookinfo、tb_reader 

管理 员 登 录 后 ， 选 择 “图 书 借 还 ”/“ 图 书 续 借 ”菜单 项 ， 进 入 到 图 书 续 借 页 面 。 在 该 页 面 的 “ 读 
者 条 形 码 ”文本 框 中 输入 读者 的 条 形 码 (如 123456789) 后 ， 单 击 “确定 ”按钮 ， 系 统 会 自动 检索 出 该 
读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 则 将 其 显示 在 页 面 中 ， 此 时 
单 击 “ 续 借 ” 超 链接 , 即 可 续 借 指定 图 书 (即将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 )。 
图 书 续 借 页 面 的 运行 结果 如 图 6.31 所 示 。 


/ 
说 明 当 读 者 续 借 完 图 书后 ， 操 作 员 通 过 单机 “完成 续 借 ” 按钮 ， 将 重新 载 入 图 书 续 借 页 面 ， 
当前 页 处 于 空 信息 状态 ， 从 而 方便 操作 员 进行 下 一 个 读者 续 借 图 书 操作 。 


图 书 续 借 页 面 的 设计 方法 同 图 书 借阅 类 似 ， 所 不 同 的 是 ， 在 图 书 续 借 页 面 中 没有 添加 借阅 图 书 的 
功能 ， 而 是 添加 了 “ 续 借 ” 超 链接 。 图 书 续 借 页 面 在 Dreamweaver 中 的 设计 效果 如 图 6.32 所 示 。 
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图 6.31 图 书 续 借 页 面 的 运行 结果 图 6.32 图 书 续 借 页 面 的 设计 效果 


单 击 “ 续 借 ” 超 链接 时 ， 还 需要 将 读者 条 形 码 、 借 阅 id 号 和 图 书 归 还 时 间 一 同 传递 到 图 书 续 借 的 
处 理 页 borrow_oncemore.php 中 。 代 码 如 下 : 
倒 性 25 ”代码 位 置 ， 光盘 \TM\06\library\bookRenew.php 
<a href="borrow_oncemore.php?barcode=<?php echo S$info[barcode];?>&borrid=<?php echo 
S$info[borrid];?>&backTime=<?php echo $info[backTime];?>"> 续 借 </a> 
检索 读者 信息 和 读者 借阅 信息 的 SQL 语句 如 下 : 
倒 程 26 ”代码 位 置 ， 光盘 \TM\06\library\bookRenew.php 


$sql=mysql_query("select borr.id as borrid,borrborrowTime,borrbackTime,borrifback,r*,tname as typename, 
t.numberbook.bookname,book.price,pub.pubname,bc.name as bookcase from tb_borrow as borr join tb_reader 
r on borr.readerid=r.id join tb_readerType t on r.typeid=t.id join tb_bookinfo as book on book.id=borr.bookid join 
tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id where 
rbarcode='$barcode' and borr.ifoack=0"); 


$info=mysql_fetch_array($sql); /检索 读者 信息 和 借阅 信息 


单 击 “ 续 借 ” 超 链接 ， 提 交 到 数据 处 理 页 borrow_oncemore.php， 主 要 用 于 完成 图 书 的 续 借 功能 ， 
主要 通过 更 改 图 书 的 归还 日 期 (即将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 ， 续 借 日 期 
的 具体 算法 ， 参 见 6.9.1 节 的 详细 讲解 ) 实现 。 数 据 处 理 页 的 代码 如 下 : 

倒 程 27 ”代码 位 置 ， 光盘 \TM\06\library\borrow_oncemore.php 
<?php 


session_start(); /初始 化 session 变量 
include("conn/conn.php"); /连接 数据 库 文件 
$barcode=$_GET[barcode]; /获取 图 书 条 形 码 
$new=$_GET[backTime]; // 获 取 图 书 归还 时 间 


// 更 新 续 借 期 ， 将 动态 获取 的 还 书 期 日 转化 为 时 间 戳 ， 然 后 再 求 出 续 借 后 的 还 书 日 期 
$newbackTime=date("Y-m-d",(mktime(0, 0, 0, substr($new,5,2), substr($new,8,2), substr($new,0,4))+3600*24*30)); 


$borrid=$_GET[borrid]; 


/获取 续 借 图 书 的 id 号 


mysql_query("update tb_borrow set backTime='$newbackTime ',ifback=0,operator='$_SESSION[admin_name]' 


where id=$borrid"); 


echo "<script language=javascript>alert( 图 书 续 借 操 作成 功 ! ');window.location.href='bookRenew.php? barcode= 


$barcode'; </script>"; 
?> 


@ 


/弹出 图 书 续 借 成 功 的 提示 信息 
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6.8.5 图 书 归 还 的 实现 过 程 


国 。 图书 归 还 使 用 的 数据 表 : tb borow、tb bookinfp、tb reader 

管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 归 还 ”菜单 项 ， 进 入 到 图 书 归还 页 面 。 在 该 页 面 的 “ 读 
者 条 形 码 ” 文 本 框 中 输入 读者 的 条 形 码 (如 123456789) 后 ， 单 击 “ 确 定 ”按钮 ， 系 统 会 自动 检索 出 该 
读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检 索 到 对 应 的 读者 信息 ， 则 将 其 输出 到 浏览 器 ， 此 时 
单 击 “ 归 还 ” 超 链接 ， 即 可 将 指定 图 书 归还 。 图 书 归 还 页 面 的 运行 结果 如 图 6.33 所 示 。 
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图 6.33 图 书 归还 页 面 的 运行 结果 
图 书 归还 页 面 的 设计 方法 同 图 书 续 借 类 似 ， 所 不 同 的 是 ， 将 图 书 续 借 页 面 中 的 “ 续 借 ” 超 链接 更 
换 为 “归还 ” 超 链接 。 在 单 击 “ 归 还 ” 超 链接 时 ， 也 需要 将 读者 条 形 码 和 借阅 id 号 一 同 传递 到 图 书 归 
还 处 理 页 。 代 码 如 下 : 
倒 程 28 ”代码 位 置 ; 光盘 \TM\O6\library\bookBack.php 


<a href="bookBack_ok.php?borrid=<?php echo S$info[borrid];?>&barcode=<?php echo $info[barcode];?>"> 归 还 
</a> 


检索 读者 信息 及 读者 借阅 信息 的 SQL 语句 如 下 : 

倒 性 29 ”代码 位 置 ， 光盘 \TM\06\library\bookBack.php 

$sql=mysql_query("select borr.id as borrid,borrborrowTime,borrbackTime,borrifback,r*tname as typename, 
t.numberbook.bookname,book.price,pub.pubname,bc.name as bookcase from tb_borrow as borr join tb_reader 
r on borr.readerid=r.id join tb_readerType t on r.typeid=t.id join tb_bookinfo as book on book.id=borr.bookid join 
tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id where 
r.barcode='$barcode' and borr.ifback=0"); 

$info=mysql_fetch_array($sql); /检索 读者 信息 及 该 读者 的 借阅 信息 


单 击 “ 归 还 ” 超 链接 ， 即 可 指定 图 书 归还 。 数 据 处 理 页 的 代码 如 下 : 
倒 程 30 ”代码 位 置 ， 光盘 \TM\06\library\bookBack_ok.php 


<?php 
session_start(); /| 初始 化 session 变量 
include("conn/conn.php"); // 连 接 数 据 库 文件 


$backTime=date("Y-m-d"); // 归 还 图 书 日 期 
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$borrid=$_GET[borrid]; 
mysql_query("update tb borrow set backTime='$backTime ',ifback=1,operator='$_SESSION[admin_name] 
where id=$borrid"); 
echo "<script language='ijavascript>alert(' 图 书 归还 操作 成 功 ! ');window.location.href='bookBack.php?barcode= 
// 弹 出 图 书 归 还 成 功 的 提示 信息 


$barcode';</script>"; 


?> 


/获取 读者 的 id 号 
/更 新 读者 的 借阅 信息 


6.8.6 ”图书 借阅 查询 的 实现 过 程 


国 ” 图书 借 阅 查询 使 用 的 数据 表 : tb_borrow、tb_bookinfo、tb_reader 
管理 员 登 录 后 ， 选 择 “ 系 统 查询 ”/“ 图 书 借阅 查询 ”菜单 项 ， 进 入 到 图 书 借阅 查询 页 面 。 图 书 借 
阅 查 询 页 面 的 运行 结果 如 图 6.34 所 示 。 在 该 页 面 中 可 以 按 指定 的 字段 或 某 一 时 间 段 进行 查询 ， 同 时 还 


可 以 实现 按 指定 字段 及 时 间 段 进行 综合 条 件 查 询 。 


图 书 借阅 


狼 二 上 襄 图 隶 馆 管理 系统 


SE 
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EE 


oogEiaht 和 2007 vee nrteel cm 吉林 se 大池 相思 


可 让 再 人 用 IE 8 0 区 QL 上 六 40s4rm6a 和 归 信 旺 未 井 其 


图 6.34 图书 借阅 查询 页 面 的 运行 结果 


入 的 查询 条 件 是 否 合法 。 该 页 面 中 所 涉及 的 表单 元 素 如 表 6.5 所 示 。 


表 6.5 图 书 借阅 查询 页 面 所 涉及 的 表单 元 素 


查询 页 面 主要 用 于 收集 查询 条 件 和 显示 查询 结果 , 并 通过 自 定义 的 JavaScript 函数 验证 输 


名 称 元 素 类 型 重要 属性 含义 
myform form method="post" action="" 表 
flagl checkbox value="a" 请 选择 查询 依据 
flag2 checkbox value="b" 借阅 时 间 
<option value="k.barcode" > 图 书 条 形 码 </option> 
本 <option Vahe= bookaame ”图 书 名 op 查询 字段 
<option value="rbarcode"> 读 者 条 形 码 </option> 
<option value="rname"> 读 者 名 称 </option> 
key text size="50" 关键 字 
sdate text id="sdate" 开始 日 期 
edate text id="edate" 结束 日 期 
Submit submit onClick="retum check(myform):" “查询 ”按钮 
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在 图 书 借 阅 查询 页 面 中 ,指定 查询 条 件 后 ， 提 交 表 单 信息 到 当前 页 。 首 先 获 取 表 单元 素 复 选 框 flag 
的 值 ， 然 后 根据 flag 的 值 组 合 查 询 字符 串 。 

如 果 flagl 的 值 等 于 a， 那 么 按 指 定 的 字段 检索 图 书 借阅 信息 ; 如 果 flag2 的 值 等 于 b， 那 么 按 指 定 
的 时 间 段 检索 图 书 借阅 信息 ; 如 果 flagl 的 值 等 于 a， 并 且 flag2 的 值 等 于 b， 那 么 按 以 上 两 个 条 件 的 综 
合 条 件 检索 图 书 借 阅 信息 ， 并 将 查询 结果 输出 到 浏览 器 。 具 体 代码 如 下 : 

全 性 31 ”代码 位 置 ， 光盘 \TM\06\library\borrowQuery.php 

tt /| 连接 数据 库 文件 


$sql=mysql_query("select b.borowTime,b.backTime,b.ifback,r.barcode as readerbarcode,r.name,k.id,k.barcode, k.bookname 
from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id"); // 查 询 图 书 借阅 信息 


if($_POST[ISubmit]="){ // 如 果 提 交 了 表单 ， 则 执行 以 下 操作 
$f=$_POSTIf; // 获 取 操 作 员 选 择 的 查询 条 件 
$key1=$_POST[key1]; /获取 查询 关键 字 
$sdate=$_POST[sdate]; // 获 取 借 阅 的 起 始 日 期 
$edate=$_POST[edate]; /获取 借阅 的 结束 日 期 
S$flag1=$_POST[fag1]; // 获 取 按 指定 条 件 查 询 的 复 选 框 值 
$flag2=$_POSTI[flag2]; /| 获取 按 日 期 查询 的 复 选 框 值 
if($flag1=="a"){ // 如 果 按 指定 条 件 查询 ， 则 执行 以 下 语句 


$sql=mysql_query("select b.borrowTime,b.backTime,bjifback,rbarcode as readerbarcode,r.name,k.id,k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=rid join tb_bookinfo k on b.bookid=k.id where $f like 
'%$key1%"); 


} 

if($flag2=="b"){ // 如 果 按 时 间 段 查询 ， 则 执行 以 下 语句 

$sql=mysql_query("select b.borowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name, k.id, k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id where 
borrowTime between '$sdate' and '$edate'"); 


} 

if($flag1=="a" && $flag2=="b"}{ // 如 果 按 综合 条 件 查 询 ， 则 执行 以 下 语句 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=rid join tb_bookinfo k on b.bookid=k.id 
where borrowTime between '$sdate' and '$edate' and $f like '%$key1%"); 

} 


} 
$result=mysql_fetch_array($sql); // 检 索 查 询 结 果 
if($result==false){ // 如 果 查询 结果 不 存在 ， 则 弹出 提示 信息 
Tp 
<table width="100%" height="30" border="0" cellpadding="0" cellspacing="0"> 

<tr> 

<td height="36" align="center"> 暂 无 图 书 借阅 信息 ! </td> 

</tr> 
</table> 
<?php 


下 
else{ /否则 ， 输 出 图 书 借阅 信息 
ye 
<table width="723” border="1" cellpadding="0" cellspacing="0" bordercolor="#FFFFFF" bordercolordark= "#D2E3E6" 
bordercolorlight="#FFFFFF"> 
<tr align="center bgcolor="#D0OE9F8"> 
<td width="13%"> 图 书 条 形 码 </td> 
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<td width="27%"> 图 书 名 称 </td> 
<td width="15%"> 读 者 条 形 码 </td> 
<td width="11%"> 读 者 名 称 </td> 
<td width="13%"> 借 阅 时 间 </td> 
<td width="11%"> 归 还 时 间 </td> 
<td width="10%"> 是 否 归还 </td> 
</tr> 
<?php 
do{ 
if($result[ifback]=="0"){ // 如 果 “ 是 否 归 还 ”等 于 0, 则 输出 “未 归还 ” 
$ifbackstr=" 未 归还 "; 
' 
else{ // 如 果 “ 是 否 归还 ”等 于 1, 则 输出 “已 归还 ” 
$ifbackstr=" 已 归还 "; 
和 
?> 
ss 出 符合 查询 条 件 的 记录 人 setrersrrtinstss 
<tr> 
<td style="padding:5px;">&nbsp;<?php echo $result[barcode];?></td> 
<td style="padding:5px;"><a href="book_look.php?id=<?php echo Sresultlid]; ?>"><?php echo $result 
[bookname]; ?></a></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[readerbarcode];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[name];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[borrowTime];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[backTime];?></td> 
<td style="padding:5px;">&nbsp;<?php echo S$ifbackstr;?></td> 
</tr> 
A 
<?php 
}while($result=mysql_fetch_array($sql)); 


?> 


6.8.7 ”单元 测试 


在 开发 完 图 书 借 还 模块 后 ， 对 该 模块 进行 了 单元 测试 。 当 操作 员 按 指定 字段 进行 条 件 查询 时 ， 能 
够 实现 图 书 借阅 查询 ， 当 操作 员 按 指定 的 时 间 段 进行 查询 时 ， 也 能 够 实现 图 书 借阅 查询 ， 但 是 当 操 作 
员 按 以 上 两 个 条 件 同时 进行 查询 时 ， 则 不 能 实现 图 书 借阅 查询 。 

图 书 借 还 模块 的 表单 元 素 代码 如 下 : 


<input name="flag" type="checkbox" class="noborder" value="a" checked> 
<input name="flag" type="checkbox" class="noborder" id="flag" value="b"> 


完成 图 书 借阅 查询 的 SQL 语句 如 下 : 
if($flag=="a"){ 
$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, k.barcode, 


k.bookname from tb_borrow b join tb_reader r on b.readerid=rid join tb_bookinfo k on b.bookid=k.id where $f like 
'%$key1%"); 


@ 
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} 
if($flag=="b")}{ 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id where 
borrowTime between '$sdate' and '$edate'"); 
} 
if($flag=="a" && $flag=="b"){ 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifoack,r.barcode as readerbarcode,r.name,k.id,k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id where 
borrowTime between '$sdate' and '$edate' and $f like '%$key1%"); 
2 


为 了 找 出 错误 的 原因 ， 笔 者 应 用 echo0 语 句 对 $flag 的 值 进行 输出 ， 输 出 的 结果 显示 $flag 等 于 b， 
不 执行 $flag 等 于 a。 这 充分 说 明 在 对 同名 复 选 框 进行 查询 时 ， 它 只 能 获取 最 后 一 个 复 选 框 的 值 。 

解决 该 问题 的 方法 是 将 两 个 复 选 框 分 别 定义 为 两 个 不 同 的 名 称 , 然后 赋予 不 同 的 值 ， 再 进行 判断 ， 
就 可 以 实现 按 指定 字段 、 指 定 的 条 件 进行 综合 条 件 查 询 了 。 

因此 将 表单 中 的 HTML 标记 中 的 flag 进行 重 命名 ， 代 码 如 下 : 


<input name="flag1" type="checkbox" class="noborder" value="a" checked> 
<input name="flag2" type="checkbox" class="noborder" id="flag" value="b"> 


修改 后 的 完成 图 书 借阅 查询 的 SQL 语句 如 下 : 


if($flag1=="a")}{ 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id,k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id where $f 
like '%$key1%"); 
下 
if($flag2=="b"}{ 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifoack,r.barcode as readerbarcode,r.name,k.id,k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id where 
borrowTime between '$sdate' and '$edate'"); 
} 
if($flag1=="a" && $flag2=="b"){ 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifoack,r.barcode as readerbarcode,r.name,k.id,k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id where 
borrowTime between '$sdate' and '$edate' and $f like '%$key1%"); 
} 


6.9 开发 技巧 与 难点 分 析 


6.9.1 如 何 自动 计算 图 书 归 还 日 期 


在 图 书馆 管理 系统 中 会 遇 到 这 样 的 问题 : 在 借阅 图 书 时 ， 需 要 自动 计算 图 书 的 归还 日 期 。 
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1. 图 书 归 还 日 期 


根据 图 书馆 还 书 的 规律 一 般 都 以 30 天 为 一 个 期 限 ， 因 此 在 图 书 归 还 时 ， 可 以 设置 一 个 固定 的 值 ， 
即 30 天。 计算 归还 日 期 的 方法 如 下 : 

图 书 归还 日 期 = 系统 当前 日 期 + 借阅 天 数 固定 值 30 天 。 

自动 计算 图 书 归还 日 期 的 具体 代码 如 下 : 


date("Y-m-d",(time()+3600*24*30)) // 图 书 归还 日 期 


2. 续 借 图 书 归还 日 期 


续 借 图 书 归还 日 期 是 在 原来 数据 库 保存 该 图 书 归 还 日 期 (这 个 日 期 是 不 固定 的 ) 的 基础 上 再 次 借阅 
所 计算 的 时 间 ， 它 是 需要 根据 数据 表 中 保存 的 归还 日 期 来 计算 的 。 计 算 图 书 续 借 归还 日 期 的 方法 如 下 : 

续 借 图 书 归 还 日 期 = 所 借 图 书 在 数据 表 中 的 归还 日 期 + 借阅 天 数 固定 值 30 天 。 

首先 应 用 substrO) 函 数 分 别 取 出 所 借 图 书 在 数据 表 中 原 定 的 归还 日 期 “月 入 “日 “年 ” 然后 应 
用 mktime() 函 数 计算 出 归还 日 期 的 时 间 戳 ,最 后 应 用 date() 函 数 格 式 化 日 期 为 “YYYY-MM-DD” 格 式 。 
自动 计算 续 借 图 书 归 还 日 期 的 代码 如 下 : 


$new=$_GET[backTime]; /获取 传递 过 来 的 该 图 书 在 数据 表 中 的 归还 日 期 
// 更 新 续 借 期 ， 将 动态 获取 的 还 书 日 期 转化 为 时 间 戳 ， 然 后 再 求 出 续 借 后 的 还 书 日 期 
date("Y-m-d",(mktime(0, 0, 0, substr($new,5,2), substr($new,8,2), substr($new,0,4))+3600*24*30)); 


6.9.2 ”如 何 对 图 书 借阅 信息 进行 统计 排行 
在 图 书馆 管理 系统 的 首页 中 ， 提 供 了 显示 图 书 借阅 排行 榜 功 能 。 要 实现 该 功能 ， 最 重要 的 是 如 何 


获取 统计 排行 信息 ,这 可 以 通过 一 条 SQL 语句 实现 ,本 系统 中 实现 对 图 书 借阅 信息 进行 统计 排行 的 SQL 
语句 如 下 : 
select * from (select bookid,count(bookid) as degree from tb_borrow group by bookid) as borr join (select b.*, 
c.name as bookcasename,p.pubname,ttypename from tb_bookinfo b left join tb_bookcase c on b.bookcase= 


C.id join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as book on borr. 
bookid=book.id order by borr.degree desc limit 10 


下 面 将 对 该 SQL 语句 进行 分 析 : 
(1) 对 图 书 借阅 信息 表 进 行 分 组 并 统计 每 本 图 书 的 借阅 次 数 ， 然 后 使 用 as 为 其 指定 别名 为 borr。 
代码 如 下 : 


(select bookid,count(bookid) as degree from tb_borrow group by bookid) as borr 


(2) 使 用 左 联接 查询 出 图 书 的 完整 信息 ， 然 后 使 用 as 为 其 指定 别名 为 book。 代 码 如 下 : 


(select b.*,c.name as bookcasename,p.pubname,ttypename from tb_bookinfo b left join tb_bookcase c on 
b.bookcase=c.id join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as book 


(3) 使 用 join on 语句 将 borr 和 book 连接 起 来 ， 再 对 其 按 统计 的 借阅 次 数 degree 进行 降序 排序 ， 
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并 使 用 limit 子 句 限制 返回 的 行 数 。 


6.10 ”联接 语句 技术 专题 


在 实际 网 站 开发 过 程 中 ， 经 常 需要 从 多 个 表 中 查询 信息 ， 在 MySQL 数据 库 中 可 以 通过 连接 的 方 
式 实现 多 表 查 询 ， 连 接 方 式 分 为 内 联接 和 外 联接 两 种 。 下 面 对 这 两 种 连接 方式 进行 详细 讲解 。 


6.10.1 内 联接 语句 


inner join 即 内 联接 查询 方式 ， 是 程序 开发 中 常用 的 连接 方式 。 内 联接 称 为 相等 联接 ， 它 返回 两 个 
表 中 的 所 有 列 ， 但 只 返回 在 联接 列 中 具有 相等 值 的 行 。 内 联接 查询 的 语法 格式 如 下 : 
select fieldlist 


from table1 [inner] join table2 
on table1.column=table2.column 


参数 说 明 如 下 。 

fieldlist， 要 查询 的 字段 列表 。 

table1、table2: 为 要 连接 的 表 名 。 

inner: 可 选项 ， 表 示 表 之 间 的 连接 方式 为 内 联接 。 

on tablel.column=table2.column: 用 于 指明 表 tablel 和 表 table2 之 间 的 连接 条 件 。 

下 面 通过 内 联接 方式 实现 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 。 代 码 如 下 : 


$sql=mysql_query("select tb_yg.userid,tb_yg.name,tb_yg.sex,tb_yg.age,tb_yg.tel,tb_yg.bm,tb_yg_info.gz from 
tb_yg inner join tb_yg_info on tb_yg.userid=tb_yg_info.ygid"); 
$info=mysql_fetch_array($sql); 


6.10.2 ”外 联接 语句 


内 联接 返回 的 是 两 个 表 中 符合 条 件 的 数据 ， 而 外 联接 返回 部 分 或 全 部 匹配 行 ， 这 主要 取决 于 所 建 
立 的 外 联接 的 类 型 。 外 联接 分 为 左 外 联接 和 右 外 联接 ,下面 对 这 两 个 外 联接 的 使 用 方法 进行 详细 讲解 。 
1. 左 外 联接 (left outer join) 
左 外 联接 返回 的 查询 结果 包含 左 表 中 的 所 有 符合 查询 条 件 及 右 表 中 所 有 满足 连接 条 件 的 行 。 
MySQL 数据 库 中 使 用 左 外 联接 的 语法 格式 如 下 : 
select field 1[field2…] 
from table1 left [outer] join table2 
on join_condition 
[where search_condition] 
参数 说 明 如 下 。 
left outer join: 表示 表 之 间 通 过 左 外 联接 方式 相互 连接 ， 也 可 以 简写 成 left join。 
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on join_condition: 指 多 表 建 立 连 接 所 使 用 的 连接 条 件 。 
where search_condition: 可 选项 ， 用 于 设置 查询 条 件 。 
下 面 通过 左 外 联接 的 方式 建立 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 。 代 码 如 下 : 


$sql=mysql_query("select * from tb_yg left outer join tb_yg_info on tb_yg.userid=tb_yg_info.ygid ",$conn); 
$info=mysql_fetch_array($sql); 


2. 右 外 联接 (right outerjoin) 

右 外 联接 返回 的 查询 结果 包含 左 表 中 的 所 有 符合 连接 条 件 以 及 右 表 中 所 有 满足 查询 条 件 的 行 。 
MySQL 数据 库 中 使 用 右 外 联接 的 语法 格式 如 下 : 

select field 1[field2…] from table1 right [outer] join table2 on join_condition [where search_condition] 


参数 说 明 如 下 。 

right outer join: 表示 表 之 间 通 过 右 外 联接 方式 相互 连接 ， 也 可 以 简写 成 right join。 

outer: 可 选项 ， 表 示 表 之 间 的 联接 方式 为 完全 联接 。 

on join_condition: 指 多 表 建 立 联接 所 使 用 的 连接 条 件 。 

where search_condition: 可 选项 ， 用 于 设置 查询 条 件 。 

下 面 通过 右 外 联接 的 方式 建立 员工 信息 表 和 员工 工资 表 的 连接 ， 并 显示 查询 结果 。 代 码 如 下 : 
$sql=mysql_query("select * from tb_yg right outer join tb_yg_info on tb_yg.userid=tb_yg_info.ygid "); 
$info=mysql_fetch_array($sql); 


6.11 本 章 总结 


本 章 运 用 软件 工程 的 设计 思想 ， 通 过 一 个 完整 的 图 书馆 管理 系统 引导 读者 深入 了 解 系统 的 开发 流 
程 。 在 这 个 系统 的 实现 过 程 中 ， 除 了 应 用 一 些 基本 的 PHP 技术 之 外 ， 还 涉及 一 些 独特 的 技术 细节 ， 如 
权限 设置 、 多 表 查 询 技术 等 需要 读者 掌握 ， 并 能 在 实际 的 操作 中 灵活 应 用 ， 举 一 反 三 。 
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如 果 有 人 问 “ 什 么 是 Web 2.0? ”， 丽 怕 没 几 个 人 能 说 得 清楚 ， 但 
是 要 问 什么 是 博客 ， 哪 怕 是 从 不 上 网 的 人 也 是 耳熟能详 的 了 。 博 容 正 
是 Web 2.0 概念 中 重要 的 组 成 部 分 之 一 (大 家 熟知 的 还 包括 IM 即时 
通 和 RSS 阅读 器 )。 

Blog (博客 )， 全 名 \Weblog， 后 来 缩写 为 Blog。Blogger 就 是 写 
Blog 的 人 ， 习 惯 于 在 网 上 写 出 日 记 、 发 布 个 人 照片 、 展 示 个 性 自我 的 
用 户 群 体 。 对 于 Blog/Blogger 的 中 文 名 称 ， 有 翻译 成 “博客 "， 也 有 
翻译 为 “网 志 "， 但 大 多 数 人 都 已 经 认可 了 “博客 "。 

通过 阅读 本 章 ， 可 以 学 习 到 : 


ml 


各 吾 吾 至 


博客 管理 系统 的 开发 流程 

进一步 掌握 如 何 做 项 目 需求 分 析 与 系统 设计 

实现 一 个 简单 的 公告 栏 管理 模块 

掌 栓 不 同 的 图 片上 传 技术 

掌握 一 种 采用 JavaScript+CSS 技术 来 实现 的 半 透 明 动 态 下 拉 菜 单 
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71. 汗 发 背 又 


博客 管理 系统 给 人 们 提供 抒发 个 人 情感 、 人 与 人 之 间 进行 良好 沟通 的 平台 ， 博 客 拥有 真实 的 内 容 ， 
可 以 通过 博客 记录 工作 、 学 习 、 生 活 和 娱乐 的 点 滴 ， 以 及 发 表 文 章 和 评论 ， 从 而 在 网 上 建立 一 个 完全 
属于 自己 的 个 人 天 地 ， 成 为 当今 网 络 最 为 个 性 化 和 平民 化 的 个 人 展示 空间 。 对 于 网 民 ， 只 要 拥有 博客 ， 
就 可 以 超越 现实 生活 ， 拥 有 不 同 以 往 的 全 新 网 上 生活 ;对 于 博客 服务 商 ， 则 必须 从 功能 提供 转型 到 全 
方位 社会 服务 的 提供 ， 建 立 虚拟 社会 ， 并 负责 维护 运行 ， 保 证 博客 日 党 生活， 对 于 社会 而 言 ， 有 利于 
构建 和 谐 的 互联 网 空间 ， 维 护 和 谐 的 社会 环境 。 从 这 个 角度 来 说 ， 构 建新 生活 方式 ， 将 是 互联 网 发 展 
的 一 个 里 程 碑 。 


7.2 需求 分 析 


信息 时 代 的 今天 ， 博 客 已 经 成 为 一 种 新 的 生活 方式 。 在 网 络 中 构建 一 个 赋 有 个 性 化 的 个 人 博客 ， 
提供 了 一 种 可 信任 的 和 实时 连通 的 网 络 环境 ， 通 过 网 络 开放 性 和 交互 性 的 特点 ， 让 用 户 在 任何 时 间 、 
任何 地 点 ， 通 过 网 络 方便 地 “生活 ”， 不 仅 是 信息 传递 与 获取 ， 还 可 以 进行 群体 交流 和 资源 共享 ， 展 示 
自我 ， 为 个 人 发 展 带 来 新 机 遇 。 

通过 对 多 个 博客 网 的 调查 分 析 ， 客 户 要 求 本 博客 管理 系统 具有 以 下 功能 
要 求 系统 采用 B/S 架构 ， 实 现 人 机 交互 。 
要 求 系统 界面 个 性 化 ， 色 彩 搭配 和 谐 ， 具 有 很 强 的 视觉 冲击 力 ， 操 作 简 便 。 
要 求 突出 主题 ， 显 示 最 新 文章 和 公告 。 
要 求 游客 可 以 浏览 文章 、 浏 览 图 片 、 发 表 评论 。 
要 求 具 有 强大 的 搜索 查询 功能 ， 实 现 精确 查询 和 模糊 查询 。 
完善 的 文章 管理 功能 ， 包 括 文章 的 发 表 、 删 除 ， 以 及 对 文章 的 评论 与 回复 。 
支持 图 片上 传 功能 ， 可 以 上 传 各 种 类 型 的 图 片 。 
支持 好 友 功 能 。 
系统 运行 稳定 ， 安 全 可 靠 。 


加 


办 办 办 办 办 多 多 提 


7.3 系统 设计 


7.3.1 系统 目标 


该 系统 主要 实现 如 下 目标 : 

回 ”系统 采用 B/S 架构 ， 实 现 人 机 交互 。 

回 ”系统 界面 设计 以 浅 色 为 主 ， 美 观 友 好 ， 操 作 简 便 。 
回 ”突出 重点 内 容 ， 显 示 最 新 文章 。 

回 ” 非 登录 用 户 可 以 浏览 文章 、 浏 览 图 片 、 发 表 评 论 。 
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全 面 的 搜索 查询 功能 ， 包 括 精 确 查 询 和 模糊 查询 。 

完善 的 文章 管理 功能 ， 包 括 文章 的 发 表 、 删 除 ， 以 及 对 文章 的 评论 与 回复 。 
支持 图 片上 传 功能 。 

支持 好 友 功能 。 

支持 公告 栏 功能 。 

系统 运行 稳定 ， 安 全 可 靠 。 


7.3.2 系统 功能 结构 


固 轿 回 罗 加 加 


博客 管理 系统 的 功能 结构 如 图 7.1 所 示 。 


图 7.1 博客 管理 系统 功能 结构 图 
7.3.3 ”系统 功能 预览 


为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 给 出 本 系统 的 儿 个 页 面 运行 效果 图 ， 如 果 想 查 
看 完整 的 效果 图 ， 请 参见 光盘 源 程序 。 

博客 管理 系统 的 首页 如 图 7.2 所 示 , 该 页 面包 含 了 系统 大 部 分 的 功能 链接 ,包括 用 户 注册 、 用 户 登 
录 、 文章 浏览 等 。 用 户 注册 页 面 如 图 7.3 所 示 , 该 页 面 显 示 了 用 户 注册 时 需要 填写 的 资料 、 注意 事项 等 。 


图 7.2 博客 首页 〈 光 盘 \TM\O7online\index.php) 图 7.3 ”用户 注册 (光盘 \TM\07\online\Register.php) 


浏览 文章 页 面 如 图 7.4 所 示 , 该 页 面 用 于 显示 文章 及 相关 的 评论 ， 也 可 以 发 表 评论 。 发 表 文 章 页 面 
如 图 7.5 所 示 ， 该 页 面 用 于 登录 用 户 发 表 文 章 ， 包 括 文章 标题 、 文 字 编 辑 区 和 文章 内 容 。 
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Eo 
sues: 下 直到 rs [本 可 7s ane We 


et a 


ww ss 
7.4 ”浏览 文章 (光盘 \TM\07\online\article.php) 图 7.5 发 表 文 章 (光盘 \TM\07\online\file.php) 


图 片上 传 页 面 用 于 上 传 图 片 , 如 图 7.6 所 示 ， 通 过 该 页 面 用 户 可 以 将 图 片 或 照片 添加 到 博客 中 。 添 
加 好 友 页 面 如 图 7.7 所 示 ， 该 页 面 用 于 输入 用 户 好 友 的 详细 信息 ， 包 括 姓名 、 性 别 、 生 日 等 。 


7.6 图片 上传 (光盘 \TM\07\online\add_pic.php) 图 7.7 添加 好 友 (光盘 \TM\07\online\friendd.php) 
7.3.4 系统 流程 图 
博客 管理 系统 的 流程 图 如 图 7.8 所 示 。 


汇 碳 靶 沪 


图 7.8 系统 流程 图 
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7.3.5 “开发 环境 


在 开发 博客 管理 系统 平台 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows Server 2003 。 

服务 器 : Apache 2.2.8。 

PHP 软件 : PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.10.3。 
开发 工具 : Dreamweaver 8。 


因 办 办 办 办 办 办 轨 


浏览 器 : IE 6.0。 
分 辨 率 : 最 佳 效果 为 1024x768 像素 。 
2， 客户 端 
浏览 器 : 推荐 使 用 IE 6.0 及 以 上 版 本 。 
分 辩 率 : 最 佳 效果 为 1024x768 像素 。 
7.3.6 ”文件 夹 组 织 结构 和 
博客 系统 的 目录 比较 少 ， 结 构 比 较 简单 ， 主 要 有 数据 库 人 
链接 文件 目录 、css 样式 目录 、JS 脚本 目录 及 背景 图 片 目录 。 一 
文件 夹 组 织 结构 如 图 7.9 所 示 。 图 7.9 博客 管理 系统 文件 夹 组 织 结构 


7.4 数据 库 设计 


7.4.1 数据 库 分 析 


本 系统 属于 中 小 型 个 人 网 站 ， 毫 无 争议 的 ， 本 系统 采用 的 依然 是 PHP+MySQL 这 对 黄金 组 合 ， 无 
论 是 从 成 本 、 性 能 、 安 全 上 考虑 ， 还 是 从 易 操 作 性 上 考虑 ，MySQL 都 是 最 佳 选择 。 


7.4.2 数据库 概 念 设计 


通过 需求 分 析 和 功能 上 的 设计 ， 本 系统 规划 出 用 户 信息 实体 、 上 传 图 片 实体 、 朋 友 圈 实体 、 文 章 
实体 和 留言 实体 。 下 面 给 出 主要 的 实体 及 E-R 图 。 

用 户 信息 实体 包括 注册 用 户 的 详细 个 人 信息 ， 如 果 想 在 本 系统 中 进行 发 表 文章 、 上 传 图 片 等 操作 ， 
则 必须 要 先进 行 注册 。 用 户 信息 实体 E-R 图 如 图 7.10 所 示 。 

上 传 图 片 实体 主要 包括 图 片 名 称 、 图 片 id、 上 传 用 户 、 上 传 室 间 和 图 片 空间 等 , 实体 E-R 图 如 图 7.11 
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所 示 。 


用 户 账号 注册 用 户 | 一 《再 户 性 出 》 D>, 


i 图 片 id 下 传 时 间 
出 em ET 


图 7.10 用 户 信息 实体 E-R 图 图 7.11 上 传 图 片 实体 E-R 图 
7.4.3 数据库 物 理 结构 设 计 


根据 实体 E-R 图 和 本 系统 的 实际 情况 ， 需 要 创建 6 张 数据 表 ， 如 图 7.12 所 示 。 
下 面 来 具体 看 一 下 这 6 张 数据 表 的 结构 设计 。 


图 服务 器 : localhost ”局 数据 库 : db_tmlog 


1. tb_user (用 户 列表 ) 于 村 六 明 
tb_articlo MyISAM 9b2312_chineso_ci 。 文章 列表 
用 户 列表 主要 存储 用 户 的 个 人 信息 。tb_user 表 的 结构 如 图 7.13 wee IE 下 人 ee 和 
所 示 tb_public MyISAM gb2312_ohines9_ci 。 公告 列表 
bd th_tpsc WyISAM gh2312_chinese_ci 。 图片 列表 


th user WNISAM 9b2312_chinese_cl 用户 列 地 


图 7.12 数据 表 列 表 


2. tb_article (文章 列表 ) 


文章 列表 存储 的 是 用 户 发 表 过 的 文章 信息 。tb_article 表 的 结 
构 如 图 7.14 所 示 。 


园 服务 器 :localhost ， 轧 数据 库 : db_tmlog ， 回 表 :tb_user 
放松 Ra 菇 名 


a al RA 说明 
机 nea0) 否 Eid 
erame varthaQm on212chinesed 百 用 记 用 全 
Tegrealname 。 verchar(20) 9b2312_ehinese_ei 下 真实 抱石 
regpwd varrnartm gp2al2_ehmasetl 百 用 P 棕 到 
veobirthday da 理 。 000c0000 用户 生 日 
Togomail vershar100)》 后 2312_ehinaso_el 否 Ti 
Tegcmy varehar lo0) 902312_ehinese_el 于 所 在 湖 市 
regico varchar(s0) ob2312_chinese_ei 和 理 用 户头 从 图 服务 器 : lecalhost ， 局 数据 库 : db_tmlog ， 国 表 :tb_article 
rageex verhart)》。 go2312_ehnese_el 。 克 用 PP 人 #8 整理 是 Null Ri 说 明 
roon varchartt0) 092312_ehinasa_el 。 要 用 Pm 储 
reghomepoge varharti00) ab2312 chinese dl。 否 用 Ps 页 i 区 多 过 
regsigm varehar200) 的 2312_ehinase_el 天 个 人 答 名 ed 藉 文章 奈 是 
revintroduce mediumtet -002312_chinese_ri 否 2 i ] | [A 本 文章 内 容 
让 verhaa20)。 qo2312 thineae el。 否 San author varchar20 。 gb2312_chnese_cl 。 再 文章 作者 
四 ne) 否 。 en pow dateime 和 要 ooo0-00-00000000 。 上 入 时 间 

图 7.13 用 户 列表 结构 图 7.14 文章 列表 结构 


3. tb_filecomment (评论 列表 ) 


评论 列表 存储 的 是 用 户 对 文章 的 评论 ， 包括 注册 用 户 和 游客 都 可 以 发 表 评 论 。tb_filecomment 表 的 
结构 如 图 7.15 所 示 。 


4. tb_tpsc (图 片 列表 ) 


图 片 列表 存储 的 是 上 传 图 片 的 信息 , 如 图 片 名 称 、 上 传 用 户 、 上 传 时 间 等 。 tb_tpsc 表 的 结构 如 图 7.16 
所 示 。 


全 
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困 服务 器 :localhost ， 曲 数据库: db_tmlog ， 园 表 : tb_filecomment 图 服务 器 : localhost 加 数据 库 : db_ tmlog ， 国 表 :tb_tpsc 

| 属 ee 量 RW bd 3 关 型 整理 尾 性 Nul 。 歌 认 党 明 
u mtg) 至 a ng) 再 目 动 号 
Tieid it) 再 0 Ee tpmc varchar(30) 。 gb2312_chinese_cl 否 图 片 名 称 
usemame varcharo) gb2312_cnmese cl 理 评论 用 户 fe mediumblob BNARY 再 二 进 制图 片 
content text 9b2312 chnese cl 到 评论 内 容 author varchar(20) 9b2312_chinese_ci 香 上 传 用 户 
datetime datetime 否 。 0000-00-90000000 。 评论 时 间 scsj date 否 。 0000-00-00 。 上 信 时 间 

图 7.15 评论 列表 结构 图 7.16 图 片 列表 结构 


5. tb_friend (好 友 列 表 ) 

好 友 列 表 主要 记录 了 姓名 、 性 别 、 生 日 等 好 友 的 个 人 信息 。tb_friend 表 的 结构 如 图 7.17 所 示 。 

6. tb_public (公告 列表 ) 

公告 列表 主要 记录 了 网 站 情况 、 博 客 系统 的 版 本 情况 或 是 网 站 活动 等 。 公 告 列表 的 结构 如 图 7.18 
所 示 。 


图 服务 器 ; localhost ， 印 数据 库 : db_tmlog ， 回 表 :由 _friend 
了 有 类 型 整理 尾 星 Nul 。 歌 认 阐明 


a nk) 和 昌 动 注 续 
ame vatcharsD) gb3912_chiness_ol 否 如 友 站 各 
sex watehardD) «002312_chinesa_o 否 本 到 性 列 
bir date 否 0000-00.00 好友 生 日 
om vareharls0) go2312_chinese_ cl 本 所 在 因 布 
ee 古 本 有 和 图 服务 器 : localhost 加 数据 库 : db_tmlog ， 回 表 :tb_public 
posteode 。 varenarl 902312_chineso ol 香 部 政 汪 码 
mail varchars0) gb2312_chinese_cl 理 E-mail 了 天 bd 是 性 Nu 丑 认 说 明 
地 Varenar20)。 01312_chmesa_ol 否 原生 码 他 ed 名 自动 加 号 
handset varchar(20) = gb2312_chineso_el 要 0 FH We varchen(s0) -962312_chinese_el 再 公告 主题 
oa warcharl(20) gb2312_chinese_cl 否 0 好 友 nQ content varchar(200) gb2312_chinese_ci 理 公告 内 容 
username vacnart20) 002312_chinese_ol 否 用 户 虹 小 pub_time date 理 发 布 时 间 
图 7.17 好 友 列表 结构 图 7.18 公告 列表 结构 


7.5 首页 设计 


7.5.1 首页 概述 


本 系统 首页 页 面 设计 简洁 ， 主 要 包括 以 下 3 部 分 内 容 。 

首部 导航 栏 : 包括 首页 链接 、 注 册 和 登录 模块 。 

左 侧 显示 区 : 包括 最 新 文章 、 最 新 图 片 和 系统 时 间 横 块 。 游 客 主要 通过 该 区 域 浏览 文章 、 浏 
览 图 片 及 发 表 评论 。 

主 显示 区 : 为 系统 公告 栏 ， 显 示 系 统 及 网 站 的 最 新 咨询 。 

本 案例 中 提供 的 首页 如 图 7.19 所 示 。 该 首页 在 本 书 光盘 中 的 路 径 为 \TM\07\tmlog\index.php。 


7.5.2 首页 技术 分 析 
在 首页 主 显示 区 ， 是 一 个 公告 栏 模块 。 公 告 栏 主要 用 于 公布 系统 版 本 的 更 新 或 升级 情况 、 网 站 的 


最 新 活动 安排 等 ,也 可 以 链接 一 些 用 户 的 精彩 文章 。 本 系统 的 公告 栏 模块 是 通过 <marquee> 标 签 来 实现 
的 。<marquee> 标 签 是 HTML 自 带 的 ， 也 是 初学 者 最 常用 的 公告 栏 实现 方式 。 使 用 <marquee> 标 签 ， 可 


_- 辐 
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以 实现 文字 或 图 片 的 滚动 效果 ， 增 加 了 实用 性 的 同时 ， 也 增加 了 观赏 性 。 下 面 就 来 学 习 一 下 这 个 标签 
的 常用 方法 及 属性 。 


梳子 


图 7.19 博客 管理 系统 首页 


转注 意 <marquee> 标 签 是 微软 与 网 景 等 公司 私 制 的 ， 从 来 没有 被 W3C 当 作 正式 标签 来 使 用 ， 所 
以 除了 下 浏览 器 外 ， 有 些 属 性 是 不 被 其 他 浏览 器 所 支持 的 ， 在 使 用 时 一 定 要 留意 。 


1. <marquee> 标 签 的 文字 移动 属性 


<marquee> 标 签 的 特点 就 是 可 以 使 文字 或 图 片 动 起 来 ， 在 早 些 时 候 ， 这 可 是 一 个 了 不 起 的 技术 。 随 
着 W3C 标准 的 逐渐 完善 ，<marquee> 标 签 已 经 越 来 越 少 有 用 武之 地 了 ， 但 有 些 技术 是 永远 不 变 的 ， 如 
<marquee> 标 签 中 的 属性 名 称 和 属性 值 ， 了 解 css 样式 表 和 JavaScript 脚本 语言 的 人 会 感觉 到 非常 熟悉 ， 
因为 这 些 属性 名 称 和 属性 值 都 经 常 被 提 及 和 使 用 。<marquee> 标 签 用 的 文字 移动 属性 及 说 明 如 表 7.1 所 示 。 


表 7.1 <marquee> 标 签 常用 的 文字 移动 属性 及 说 明 


属性 名 称 属 性 值 应 用 举例 
marquee 无 ee 除了 文字 外 ,还 可 以 是 <marquee> 你 好 ，PHP</marquee> 
文字 移动 属性 ， 分 别 表 示 从 右 往 左 、 从 | <marquee direction="up"> 从 下 到 上 移动 


direction leftright,up,down 


左 往 右 、 从 下 到 上 、 从 上 到 下 </marquee> 
文字 移动 方式 , 分 别 表示 沿 同一 方向 不 
behavior scroll,slide,altemate | 停 滚动 、 只 滚动 一 次 、 在 两 个 边界 内 来 


<marquee behavior="scroll"> 不 停 的 循环 
播放 </marquee> 


回 滚动 
loop 数值 1,2,3… 循环 次 数 ， 不 指定 则 表示 为 无 限 循环 


<marquee loop=5 behavior=slide> 只 循环 
滚动 3 次 </marquee> 

<marquee scrollmount=25> 我 漂 起 来 了 以 
</marguee> 


We | 二 a <marquee scrolldelay="500" scrollamount= 
Scrolldelay ”| 数值 100,200,300… | 延 时 ， 每 动 一 次 ， 停 止 的 时 间 10 体 停 走 老 -inahiueex 


scrollmount “| 数值 10.20.30… 滚动 速度 ， 数 值 越 大 ， 速 度 越 快 
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2. <marquee> 标 签 的 外 观 设置 


<marquee> 标 签 不 仅 可 以 设置 文字 移动 效果 , 还 能 设置 标签 的 外 观 。 外 观 包 括 对 齐 方式 、 标 签 底 色 、 
面积 等 。<marquee> 标 签 常用 的 外 观 属性 及 说 明 如 表 7.2 所 示 。 


表 7.2 <marquee> 标 签 常用 的 外 观 属性 及 说 明 


属性 说 明 应 用 举例 
aligi top,middle,bottom | 指定 文本 的 对 齐 方式 ”| <marquee align="middle"> 我 在 中 间 </marquee> 


<marquee bgcolor="#DEEBEF"> 看 到 了 吗 ? 


bgcolor 颜色 值 "#000000" 

/marguee> 
height | 数值 10,20,30… <marquee height="15"> 不 要 ， 太 矮 了 </marquee> 
width 数值 10,20,30… > = <marquee width="100"> 哇 ! 好 大 的 房子 啊 </marquee> 


3. <marquee> 标 签 的 其 他 属性 


除了 以 上 的 常规 设置 ，<marquee> 标 签 还 有 两 个 特殊 的 属性 ， 即 start) 和 stop0， 它 们 可 以 配合 
JavaScript 事件 产生 很 有 意思 的 效果 。 例 如 : 


onMouseOut="this.start()"; // 当 鼠标 移出 该 区 域 时 ， 开 始 滚动 
onMouseOver="this.stop()"; // 当 鼠标 进入 该 区 域 时 ， 停 止 滚动 


7.5.3 首页 的 实现 过 程 


博客 管理 系统 采用 二 分 栏 结构 ， 表 单 布局 。 具 体 实现 代码 如 下 : 
倒 程 01 ”代码 位 置 ， 光 盘 \TM\07\tmlog\index.php 


<?php 

session_start(); /开启 session 支持 
include "Conn/conn.php"; /包含 数据 库 链接 文件 
?> 


<table width="757” border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr align="right" valign="top"> 
<td height="149" colspan="2" background="images/head.jpg"> 


<!-- 上 部 导航 栏 --> 
</td> 
</tr> 
<tr> 
<td width="236" height="501" background=" images/left.jpg"> 
we <!-- 左 侧 导 航 栏 --> 
</td> 
<td width="521" height="501" align="center" background=" images/right.jpg"> 
< 二 -一 系统 公告 区 -一 一 > 
<?php 
$p_sql = "select * from tb_public order by id desc"; /从 数据 表 中 读 取 数 据 
$p_rst = mysql_query($p_sql,$link); /执行 SQL 语句 
?> 
<1 -- 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 <marquee> 标 签 开 始 ---------------------------------- 
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© <marquee onMouseOver=this.stop() style="WIDTH: 426px; HEIGHT 280px" onMouseOut=this. 
start() scrollamount=2 scrolldelay=7 direction=up> 
<?php 
while($p_row = mysql_fetch_row($p_rst)X{ /| 循环 输出 公告 标题 
?> 
© <a href="#" onclick="wopen=open(show_pub.php?id=<?php echo 
$p_row[0]; ?>",",'height=200,width=500,scollbars=no')"><?php echo $p_row[1]; ?></a><br> 
<?}?> 
</marquee> 
<! — -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 -一 一 一 一 一 一 一 一 一 - --: > 
</td> 
</tr> 
</table> 
< 代码 贴 十 


@ <marquee> 标 签 : <marquee> 标 签 中 包含 了 start 和 stop 两 个 特殊 的 属性 ， 还 有 速度 属性 scollamount 和 延迟 属性 
scrolldelay， 而 公告 栏 的 高 度 (height ) 和 宽度 ( width ) 则 是 通过 css 样式 设置 的 。 

@ <?php echo $p_row[1]; ?>: 循环 输出 公告 标题 ， 并 为 每 个 公告 添加 超 链 接 ， 这 里 用 到 了 JavaScript 脚本 的 open() 
方法 ， 该 方法 的 作用 是 打开 一 个 新 窗口 ， 同 时 可 以 对 窗口 样式 做 出 设置 。open() 方 法 的 一 般 格式 为 : open(" 链 接 的 url"," 
自 定义 名 称 "," 窗 口 样式 ")。 


7.6 文章 管理 模块 设计 


7.6.1 文章 管理 模块 概述 


对 一 个 博客 系统 来 说 ， 文 章 管理 是 最 基本 的 功能 ， 但 同时 也 是 最 复杂 的 一 个 功能 。 本 系统 的 文章 
管理 模块 包括 “添加 博客 文章 “查找 博客 文章 “管理 我 的 博客 “发 表 评 论 ”“ 删 除 文章 ”和 “ 删 
除 评论 ”等 6 大 功能 。 其 中 ， 普 通用 户 只 能 删除 自己 的 文章 及 对 文章 的 评论 ， 只 有 管理 员 才 有 权 删 除 
任何 一 篇 文章 及 回复 。 文 章 管理 模块 的 框架 如 图 7.20 所 示 。 


三 
pp ~ 
‘/ \ 
〖 
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图 7.20 文章 管理 模块 框架 图 
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7.6.2 ”文章 管理 模块 技术 分 析 


想 要 使 用 文章 管理 模块 ， 前 提 是 用 户 必须 登录 ， 匿 名 用 户 是 无 法 访问 这 些 功能 的 ， 想 要 删除 文章 
和 评论 ， 前 提 是 当前 用 户 要 么 是 管理 员 权 限 ， 要 么 是 文章 拥有 者 ， 和 否则 不 会 显示 删除 功能 的 。 这 两 方 
面 的 控制 都 需要 session 的 配合 ， 本 节 就 来 讲 一 下 session 的 应 用 及 常见 的 问题 处 理 。 

Session 的 中 文 译名 为 “会 话 ”， 是 指 用 户 从 进入 网 站 开始 ， 直 到 关闭 网 站 这 段 时 间 内 ,所 有 网 页 共 
同 使 用 的 公共 变量 的 存储 机 制 。session 比 cookie 更 有 优势 : session 是 存储 在 服务 器 端的 , 不 易 被 伪造 ; 
session 的 存储 没有 长 度 限制 ，session 的 控制 更 容易 等 。 


道 如 何 去 解 决 。 其 实 ， 大 多 数 的 错误 都 是 因为 对 session 的 配置 不 了 解 而 使 用 方法 不 正确 造成 的 ， 在 
php.ini 中 对 session 的 配置 如 表 7.3 所 示 。 
表 7.3 session 的 常用 配置 选项 
配置 选项 说 了 明 
session.save_path = c:/temp 保存 session 变量 的 目录 ， 在 Linux/UNIX 下 为 /mp 


session.ues_cookies =1 是 否 使 用 cookie 
session.name = PHPSESSID | 表示 会 话 ID 


session.auto_start = 0 是 否 自动 启用 session， 当 为 1 时 ， 在 每 页 中 就 不 必 调 用 session_start 〇 函数 了 
session.cookie lifetime =0 设 定 cookie 送 到 浏览 器 后 的 保存 时 间 ， 单 位 为 秒 。 默 认 值 为 0， 表示 直到 浏览 器 关闭 


session.cookie_path =/ cookie 有 效 路 径 

session.cookie_domain = 有 效 域名 

session.serialize handler =php | 定义 序列 化 数据 的 标识 ， 本 功能 只 有 WDDX 模块 或 PHP 内 部 使 用 ， 默 认 值 为 PHP 
Session.gc_probability =1 设 定 每 次 临时 文件 开始 处 理 的 处 理 概率 。 默 认 值 为 1 


session.gc_maxlifetime = 1440 设 定 保存 session 的 临时 文件 被 清除 前 的 存活 秒 数 
决定 参照 到 客户 端的 session 代码 是 否 要 删除 。 有 时 出 于 安全 或 其 他 考虑 ， 会 设 定 不 


session.referer_check = 删除 。 默认 值 为 0 
session.cache limiter = nocache | 设 定 session 缓冲 限制 


session.cache expire = 180 文档 有 效 期 ， 单 位 为 分 钟 
session.save_handler = files 用 于 保存 session 变量 ， 默 认 情 况 下 用 文件 


对 于 初学 者 来 说 ，session 在 php.ini 中 不 需要 特意 去 改动 ， 因 为 安装 时 会 根据 操作 系统 自行 做 出 适 
当 的 调整 。 只 在 少数 的 几 项 ， 如 session 存活 周期 (session.cookie_lifetime = 0)、 自 动 开 启 session 
(session.auto_start) 等 稍 加 改动 即 可 。 
PHP 主要 是 通过 会 话 (session) 处 理 函 数 来 对 session 进行 控制 和 使 用 的 。 常 用 的 处 理 函 数 如 表 7.4 
所 示 。 


表 7.4 PHP 常用 的 会 话 处 理 函数 


函数 函数 说 明 


session start(); | 开启 session 或 返回 已 经 存在 的 session 
$_SESSION['name'] = value; | 注册 一 个 session 变量 


session_id0 设 定 或 取得 当前 的 session_ id 值 
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续 表 
函数 函数 说 明 
检测 指定 的 session 值 是 否 存在 。isset 不 只 可 以 检测 session， 还 可 以 检测 其 他 类 型 ， 如 
isset($_ POST[mame])、isset($ GET[name']) 等 


isset($_SESSION[mame]) 


session_regenerate id0 | 更 改 session id 的 值 
Session name() | 返回 或 改变 当前 session 的 name 


unset($_SESSION['name']) | 删除 名 为 name 的 session 


session_destroy() 结束 当前 会 话 ， 删 除 所 有 session 


全 注意 (1) 如 果 要 改变 当前 session 的 name 值 ， 必 须 在 session() 之 前 调用 session_name() 函 数 ， 
而 且 session name 不 能 全 部 是 数字 ， 否 则 会 不 停 地 生成 新 的 session id。 
(2) 不 可 以 写成 unset($_SESSION)， 这 样 会 禁止 整个 会 话 的 功能 。 
在 本 节 最 后 ， 介 绍 使 用 session 时 要 注意 的 儿 个 问题 。 
1. 尽 可 能 地 将 session_start() 放 到 第 1 行 
这 种 情况 是 新 手 最 容易 犯 的 错误 。 产 生 的 错误 代码 为 : 


Warning: session_start() [function.session-start]: Cannot send session cache limiter - headers already sent… 


其 原因 就 是 在 使 用 session_start(0) 之 前 ， 就 有 HTML 代码 输出 了 。 也 许 有 的 读者 会 说 :“ 没 有 ， 
session_start() 之 前 没有 任何 代码 ， 绝 对 没有 。” 那 么 ,请 检查 你 的 程序 是 不 是 有 空 行 ， 或 类 似 echo 语句 
的 输出 。 如 果 有 ， 请 去 掉 ， 因 为 就 算是 一 个 小 小 的 空格 都 是 不 可 以 的 。 所 以 ， 为 了 避免 这 类 错误 的 发 
生 ， 尽 可 能 地 将 session_start() 放 到 第 1 行 。 


2. 在 使 用 session 之 前 一 定 要 先 写 session_start() 


大 多 数 读者 在 使 用 session 之 前 都 能 先 调用 session_startO0) 函 数 , 但 对 于 session_destroy() 函 数 却 经 常 
忽略 。session_destroy0 虽 然 是 结束 当前 会 话 并 删除 所 有 session， 但 在 删除 之 前 ， 也 要 先 开启 session 支 
持 才 可 以 ， 不 然 会 产生 这 样 的 错误 代码 : 


session_destroy() [function.session-destroy]: Trying to destroy uninitialized session in… 
所 以 ， 凡 是 在 使 用 session 或 session 函数 的 页 面 中 ， 都 要 加 上 session_start() 这 人 句 话 。 
3. 删除 所 有 session 
如 果 想 删除 所 有 session， 但 又 不 想 结束 当前 会 话 ， 用 unset 一 个 一 个 删除 实在 是 太 麻 烦 了 ,最 简单 
的 办 法 就 是 将 一 个 空 数组 赋 给 $_SESSION， 如 $_SESSION = array0， 这 样 就 解决 了 。 
7.6.3 添加 文章 的 实现 过 程 


国 添加 文章 模块 使 用 的 数据 表 : tb_article 
当 用 户 登 录 后 ， 系 统 会 直接 进入 到 文章 添加 页 (file.php)， 也 可 以 通过 单 击 “ 文 章 管理 ”/“ 添 加 
博客 文章 ” 回 到 file.php 页 。 添 加 文章 页 面 的 运行 结果 如 图 7.21 所 示 。 


@ 
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图 7.21 添加 文章 页 面 的 运行 结果 


添加 文章 页 为 一 个 发 布 表单 ， 包 括 文章 主题 、 文 字 编 辑 、 文 章 内 容 等 元 素 。 部 分 表单 元 素 如 表 7.5 


所 示 。 


名 称 
myform 
txt title 


font 
size 
color 


file 
btnt 


元 素 类 型 
form 


text 


select 


select 


select 


textarea 


submit 


表 7.5 添加 文章 页 面 的 主要 表单 元 素 


重要 属性 含义 
method="post" action="check_file.php" 添加 文章 表单 
id="txt title" size="68" 文章 标题 
name="font" class="wenbenkuang" id="font" 

的 文章 字体 
onChange="showfont(this.options[this.selectedIndex].value)" 
class="wenbenkuang" 

0 字体 大 小 
onChange="showsize(this.options[this.selectedIndex].value)" 
onChange="showcolor(this.options[this.selectedIndex].value)" 

& orthis.options[ . J.value) 字体 颜色 
name="color" size="1" class="wenbenkuang" id="select" 
cols="75" rows="20" id="file" style="border:Opx;width:520px:" 文章 内 容 
id="btn_tj" value=" 提 交 " onClick="retum checkO:" “提交 ”按钮 


当 用 户 填写 完 博客 主题 和 文章 内 容 后 ， 单 击 “ 提 交 ” 按 钮 ， 系 统 将 跳 转 到 处 理 页 (check_file.php) 
进行 处 理 。 在 处 理 页 中 ， 将 传 过 来 的 文章 标题 、 文 章 作者 和 文章 内 容 等 参数 组 成 insert 语句 ， 并 最 终 保 
存 到 数据 表 中 。 如 果 添 加 信息 成 功 ， 系 统 返 回 到 本 页 ， 可 继续 执行 添加 操作 ;如 果 添 加 失败 ， 则 返回 
到 上 一 步 。 程 序 的 关键 代码 如 下 : 

倒 程 02 代码 位 置 ， 光盘 \TM\07\tmlog\check_file.php 


<?php 


session_start(); 


include "Conn/conn.php"; /包含 数据 库 链 接 文件 


/开启 session 支持 


if($btn_tj<>"™"X /| 判断 传 值 页 面 
$title=$_POST[txt_title]; // 取 得 文章 标题 
$author=$_SESSION[username]; /取得 作者 
$content=$_POSTfile]; // 取 得 文章 内 容 
$now=date("Y-m-d H:i:s"); /使 用 date() 函 数 生成 发 布 时 间 


/* 生 成 insert 语句 */ 

$sql="Insert Into tb_article (title,content,author,now) Values (Sititle',$content,$author,$now')"; 

$result=mysql_query($sql); /执行 insert 语句 

/根据 $result， 返 回 结果 */ 

if($result)f 
echo "<script>alert( 恭 喜 您 ， 你 的 文章 发 表 成 功 山 );window.location.href=Yile.php';</script>"; 


@ 
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else{ 

echo "<script>alert( 对不起， 添加 操作 失败 山 );history.go(-1);</script>"; 
} 
}else{ 

echo "<script>alert( 对不起， 添加 操作 失败 山 );history.go(-1);</script>"; 
} 
i 


> 
说 明 文章 添加 页 面 中 ， 使 用 了 部 分 UBB 语法 ， 由 于 UBB 语法 不 属于 本 书 的 范畴 ， 所 以 这 里 
不 做 讲解 ， 请 感 兴趣 的 朋友 查看 相关 的 书籍 。 


7.6.4 文章 列表 的 实现 过 程 


国 ” 查 看 文章 列表 使 用 的 数据 表 : tb_article 
单 击 “文章 管理 ”/“ 我 的 文章 ”， 将 显示 用 户 发 表 过 的 文章 列表 。 文 章 列表 页 面 (myfiles.php) 的 
运行 结果 如 图 7.22 所 示 。 
Fe 
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图 7.22 文章 列表 页 面 的 运行 结果 
文章 列表 页 面 使 用 了 分 页 技术 和 do…while 循环 语句 来 输出 文章 标题 。 程 序 关键 代码 如 下 : 
倒 程 03 ”代码 位 置 ， 光盘 \TM\07\tmlog\myfiles.php 


<?php 
session_start(); /开启 session 支持 
include "Conn/conn.php"; /包含 数 据 库 链接 文件 
include "check_login.php"; // 包 含 权限 检查 文件 
?> 
/分 页 */ 
<?php 
/*$page 为 当前 页 ， 如 果 $page 为 空 ， 则 初始 化 为 1*/ 
if ($page=="" 
$page=1;} 
© iflis_numeric($page)){ 
$page_size=20; /| 每 页 显示 20 条 记录 
$query="select count(*) as total from tb_article where author = ".$_SESSION[usemame]." order by id 
desc"; 
$result=mysql_query($query); /查询 符合 条 件 的 记录 总 条 数 


$message_count=mysql_result($result,0,"total"); /要 显示 的 总 记录 数 
$page_count=ceil($message_count/$page_size); 。 // 根 据 记录 总 数 除 以 每 页 显示 的 记录 数 求 出 所 分 的 页 数 
S$offset=($page-1)*$page_size; // 计 算 下 一 页 从 第 几 条 数据 开始 循环 
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$sql=mysql_query("select id,title from tb_article where author = ".$_SESSION[username].”order by id 
desc limit $offset, $page_size"); 
$info=mysql_fetch_array($sql); 
?3> 
<?php 
/输出 结果 集 */ 
if($infoX{ 
$i=1; /文章 序号 
@ do{ /do…while 循环 开始 
?> 
<tr> 
<!-- 显 示 文 章 序号 和 文章 标题 -> 
<td width="498” align="left” valign="top"> &nbsp;&nbsp;&nbsp;<a href="showmy.php?file_id=<?php echo 
$infofid];?>"><?php echo $i.”、".$info[title];?></a> </td> 
</tr> 
<?php 
$i=$i+1; 
}while($info=mysql_fetch_array($sql)) // 循 环 结束 
?> 
</table></td> 
</tr> 
</table></td> 
</tr> 
<?php } ?> 


<!-- 翻 页 条 --> 

<td width="33%">&nbsp;&nbsp; 页 次 : <?php echo $page;?>/<?php echo $page_count;?> 页 &nbsp; 记 录 : <?php 
echo $message_count;?> 条 &nbsp; </td> 

<td width="67%" align="right" class="hongse01"> 

<?php 

/* 如 果 当 前 页 不 是 首页 */ 

if($page!=1){ 

/* 显 示 “ 首 页 ” 超 链接 */ 

echo "<a href=myfiles.php?page=1> 首 页 </a>&nbsp;"; 

/* 显 示 “ 上 一 页 ” 超 链 接 */ 

echo "<a href=myfiles.php?page=".($page-1)." > 上 一 页 </a>&nbsp;"; 


} 

/* 如 果 当 前 页 不 是 尾 页 */ 

if($page<$page_count) 

‘ 

/* 显 示 “ 下 一 页 ” 超 链接 */ 

echo "<a href=myfiles.php?page=".($page+1)."> 下 一 页 </a>&nbsp;"; 
/* 显 示 “ 尾 页 ” 超 链接 */ 

echo "<a href=myfiles.php?page=".$page_count."> 尾 页 </a>"; 

从 


?> 
< 人 代码 贴 十 


@ is_numeric($page): is_numeric() 函 数 判 断 变 量 Spage 是 否 为 数字 ， 如 果 是 数字 ， 则 返回 true， 否 则 返回 false。 如 
果 一 组 字符 全 部 由 数字 组 成 ， 那 么 is_numeric(0) 函 数 也 将 返回 true. 
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@ do…while 循环 和 while 循环 : do…while 循环 是 先 执行 {} 中 的 代码 段 ， 然后 判断 while 中 的 条 件 表达 式 是 否 成 立 ， 
如 果 成 立 返 回 true， 重 复 输出 人 中 的 内 容 ， 否 则 结束 循环 ， 执 行 while 下 面 的 语句 。while 循环 是 先 判 断 while 中 的 表达 
式 ， 当 返回 true 时 ， 再 执行 人 } 中 的 代码 。 两 者 的 区 别 是 : do…while 循环 比 while 循环 多 输出 1 次 结果 。 


7.6.5 查看 文章 、 评 论 的 实现 过 程 


国 ” 查 看 文章 、 评 论 使 用 的 数据 表 : tb_article、tb_filecomment 
单 击 列表 中 任意 一 个 文章 标题 , 都 会 看 到 对 应 的 文章 内 容 和 文章 评论 。 查 看 文章 页 面 (showmy.php) 
的 运行 结果 如 图 7.23 所 示 。 
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图 7.23 查看 文章 页 面 的 运行 结果 
系统 根据 当前 页 面 传 过 来 的 文章 id 值 从 数据 表 tb_article 中 返回 对 应 的 文章 信息 (包括 文章 id、 文 
章 作者 、 文 章 标题 、 文 章 内 容 和 发 表 时 间 )、 输 出 文章 信息 后 ， 开 始 查 找 表 tb_filecomment 中 fileid 字 
段 值 等 于 文章 id 的 所 有 评论 集 ， 并 通过 分 页 显示 出 来 。 显 示 文 章 页 面 (showmyphp) 的 关键 代码 如 下 : 
倒 程 04 ”代码 位 置 ， 光 盘 \TM\07\tmlog\showmy.php 


<?php 
session_start(); /开启 session 支持 
include "Conn/conn.php"; // 包 含 数据 库 链接 文件 
include "check_login.php"; // 包 含 权限 检查 文件 
Sfile_id1=$_GETIfile_id]; /取得 文章 ID 号 
$bool = false; 1/ 判断 用 户 删 除权 限 ,默认 的 false 没有 权限 
?> 
<!-- 首 部 导航 栏 --> 
<! -- -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 - 一 > 
<!-- 文 章 显示 区 -> 
<?php 
$sql=mysql_query("select * from tb_artice where id = ".$file_id1); // 根 据 文章 ID 号 查找 文章 
$result=mysql_fetch_array($sql); // 返 回 结果 集 
?> 
5 显示 文章 一 -一 一 一 一 -一 -> 


<table width="100%” border="1" cellpadding="1" cellspacing="1" bordercolor="#D6E7AS" bgcolor="#FFFFFF" 
class="i table"> 


<tr bgcolor="#FFFFFF"> 
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<td width="14%" align="center> 博 客 ID 号 </td> 

<!- 一 一 一 一 -显示 文章 id 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 > 
<td width="15%"><?php echo $result[id]; ?></td> 

<td width="11%" align="center"> 作 者 </td> 

本 显示 文章 作者 一 一 一 一 一 一 一 -一 一 一 一 一- > 


<td width="18%"><?php echo $result[author]; ?></td> 
<td width="12%" align="center"> 发 表 时 间 </td> 
< 一 一 一 一 一 一 一 一 一 一 显示 发 表 时 间 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 ~- > 
<td width="30%"><?php echo $result[now]; ?></td> 
</tr> <tr bgcolor="#FFFFFF"> 
<td align="center"> 博 客 主题 </td> 
< 一 一 一 一 一 一 一 一 文章 主题 -一 > 


<td colspan="5">&nbsp;&nbsp;<?php echo $result[title]; ?></td> 
</tr> <tr bgcolor="#FFFFFF"> 
<td align="center"> 文 章 内 容 </td> 


<td colspan=' 
<td> 


"><?php echo $result[content]; ?></td> 


<?php 
/* 判 断 登 录用 户 是 否 为 管理 员 ， 或 者 是 文章 的 作者 */ 
if($_SESSION[fig]j==1 or ($_SESSION[username] == $result[author])X{ 
/* 如 果 是 ， 就 将 $bool 设 为 true*/ 
$bool = true; 
?> 
< 一 一 | 同时 显示 “删除 ”按钮 ， 并 将 文章 id 作为 url 后 缀 一 起 传 到 处 理 页 一 一 一 一 一 一 一 一 --: > 
<a href="del_file.php?file_id=<?php echo $result[idj;?>"><img src="images/A_delete.gif" width="52" height="16" 
alt=" 删 除 博 客 文章 " onClick="return d_chk();"*></a> 
<?php } ?> 
<?php 
i /分 页 代码 部 分 略 
$sql=mysql_query("select* from tb_filecomment where fileid='$file_ id1' order by id desc limit $offset, $page_size"); 
$result=mysql_fetch_array($sql); 
tnt 根据 返回 记录 集 输出 记录 "于 
if($result==false){ 
echo "<font color=#ff0000> 对 不 起 ， 没 有 相关 评论 !</font>"; 
}else{ 
do{ 
?> 
<tr> 
<td height="57" align="center valign="top" ><table width="480” border="1" cellpadding="1" 
cellspacing="1" bordercolor="#D6E7AS5" bgcolor="#FFFFFF" class="i_table"> 
<tr bgcolor="#FFFFFF"> 
<td width="14%" align="center"> 评 论 ID 号 </td> 
<1 一 一 一 一 一 一 一 一 一 一 一 一 一 一 评论 i 一 一 一 一 一 一 一 一 一 一 一 一 - > 
<td width="15%"><?php echo $result[id]; ?></td> 
<td width="11%" align="center"> 评 论 人 </td> 
<!- 一 一 一 一 评论 人 -一 一- 一 > 
<td width="18%"><?php echo $result[username]; ?></td> 
<td width="12%" align="center"> 评 论 时 间 </td> 
ee 评论 时 间 -- 一 一 -一 -一 -一 -一 - -> 
<td width="30%"><?php echo $result[datetime]; ?></td> 
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</tr> 
<tr bgcolor="#FFFFFF"> 
<td align="center"> 评 论 内 容 </td> 


< 一 一 一 一 一 评论 内 容 -一 一 -一 一 -一 一 -一 - 一 > 
<td colspan="4"><?php echo $result[content]; ?></td> 
<td> 
<?php 
/* 如 果 $bool 为 真 ， 输 出 删除 超 链接 ， 否 则 跳 过 */ 
if ($bool){ 
gr 


<a href="del_comment.php?comment id=<?php echo $resultlid]?>"><img src="images/A_delete.gif" 
width="52" height="16" alt=" 删 除 博客 文章 评论 " onClick="return d_chk();"*></a> 
<?php }?> 
</td></tr> </table></td> </tr> 
<?php 
}while($result=mysql_fetch_array($sql)); 


?> 
</table> 
… <!-- 翻 页 功能 代码 部 分 略 一 > 


7.6.6 ”删除 文章 、 评 论 的 实现 过 程 


国 ”查看 文章 、 评 论 模块 使 用 的 数据 表 : tb_article、tb_filecomment 

在 查看 文章 评论 页 面 ， 当 系统 判定 当前 用 户 为 管理 员 或 文章 作者 时 ， 在 每 篇 文章 和 评论 的 后 面 ， 
都 将 显示 相应 的 “删除 ”按钮 。 单 击 任意 的 “删除 ”按钮 ， 系 统 会 提示 是 否 删除 ， 如 果 确 认 ， 将 跳 转 
到 处 理 页 (del_file.php 和 del_comment.php)， 完 成 删除 操作 。 

在 删除 文章 的 处 理 页 中 ， 删 除 文章 的 同时 ， 也 删除 了 该 篇 文章 相关 的 评论 。 处 理 页 首先 在 文章 列 
表 (tb_article) 中 删除 id 等 于 $file_id 的 记录 ， 如 果 没 有 可 删除 记录 ， 则 提示 失败 ， 并 返回 上 一 步 ， 如 
果 删 除 成 功 , 则 转 到 评论 列表 (tb_filecomment) 中 , 删除 所 有 该 篇 文章 的 评论 。 删 除 文章 页 (del_file.php) 
的 关键 代码 如 下 : 

倒 程 05 ”代码 位 置 ， 光盘 \TM\07\tmlog\del file.php 

<?php 


session_start(); /开启 session 支持 
include "check_login.php"; /包含 权 限 检查 文件 
include "Conn/conn.php"; /包含 数据 库 链接 文件 
$sql="delete from tb_article where id=". $file_id; /删除 文章 的 delete 语句 
$result=mysql_query($sql,$link); /执行 删除 操作 
if($result}{ 
$sql1 = "delete from tb_filecomment where fileid = ".$file_id; 。// 删 除 相对 应 的 文章 评论 
S$rst1 = mysql_query($sql1,$link); /执行 删除 评论 的 操作 
if($rst1) 
echo "<script>alert( 博 客 文章 已 被 删除 !");location='myfiles.php]';</script>"; 
else 
echo "<script>alert( 删 除 失败 !");history.go(-1);</script>"; 
} 
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else{ 
echo "<script>alert( 情 客 文章 删除 操作 失败 !);history.go(-1);</script>"; 
} 
sy 


因为 删除 文章 的 过 程 也 包含 了 删除 评论 的 过 程 ， 所 以 这 里 就 不 给 出 删除 评论 的 代码 了 。 


7.7 图 片上 传 模块 设计 
由 于 动态 网 络 编程 技术 的 诞生 ， 使 得 网 络 更 加 人 性 化 ， 为 了 能 够 和 用 户 更 好 地 互动 ， 很 多 网 站 都 
提供 了 让 用 户 上 传 图 片 的 功能 。 一 个 网 站 拥有 图 片上 传 功能 是 非常 必要 的 。 
7.7.1 图 片上 传 模块 概述 
图 片上 传 在 动态 网 页 开发 过 程 中 应 用 非常 广泛 。 如 果 有 比较 好 的 图 片 想 和 其 他 人 一 同 分 享 ， 就 可 以 


通过 图 片上 传 功能 来 实现 ， 以 增加 网 站 的 核心 竞争 力 。 本 系统 的 图 片上 传 模块 主要 实现 对 图 片 的 添加 、 
浏览 、 查 询 和 删除 操作 ， 而 对 图 片 的 删除 则 只 有 管理 员 才 有 权限 。 图 片上 传 模块 框架 图 如 图 7.24 所 示 。 


普通 用 户 超级 用 户 


图 7.24 图 片上 传 模块 框架 图 


7.7.2 图 片上 传 模块 技术 分 析 

既然 是 图 片上 传 模块 ， 显 而 易 见 本 节 的 主要 技术 就 是 上 传 图 片 功能 了 。 上 传 图 片 和 上 传 文件 的 原 
理 基 本 相同 ， 下 面 就 来 学 习 如 何 上 传 图 片 和 图 片 的 两 种 保存 方式 。 

1. 上 传 图 片 的 基本 流程 

在 网 页 中 实现 上 传 图 片 功能 的 步骤 如 下 : 

(1) 通过 <form> 表 单 中 的 file 元 素 选取 上 传 数 据 。 

使 用 file 元 素 上 传 数 据 时 注意 一 点 : 就 是 在 form 表单 中 要 加 上 属性 enctype="multipart/form-data”， 
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否则 上 传 不 了 文件 (图 片 )。 
(2) 在 处 理 页 中 使 用 $_FILES 变量 中 的 属性 判断 上 传 文件 类 型 和 上 传 文件 〈 图 片 ) 大 小 是 否 符合 
$_FILES 变量 为 系统 预定 义 变量 ， 保 存 的 是 上 传 文件 〈 图 片 ) 的 相关 属性 。 使 用 格式 如 下 : 


$_FILES[namel][property]; 


$_FILES 的 相关 属性 如 表 7.6 所 示 。 


表 7.6 $_FILES 的 相关 属性 

属 性 值 说 明 
name 上 传 文件 的 文件 名 
上 传 文件 的 类 型 
上 传 文件 的 大 小 


上 传 文件 在 服务 器 中 的 临时 文件 名 
上 传 文件 失败 的 错误 代码 


(3) 使 用 move_uploaded_file() 函 数 上 传 文件 〈 图 片 ) 或 将 文件 〈 图 片 ) 以 二 进 制 的 形式 保存 到 数 
据 库 中 。 

使 用 函数 将 文件 (图片) 保存 到 对 应 的 文件 夹 中 和 以 二 进 制 的 形式 保存 到 数据 库 中 是 上 传 文件 (图 
片 ) 的 两 种 形式 ， 稍 后 将 单独 做 介绍 。 

(4) 返回 页 面 等 待 下 一 步 操作 。 

2. 使 用 上 传 函 数 保存 文件 〈 图 片 ) 

使 用 上 传 函 数 上 传 文件 〈 图 片 ) 的 本 质 就 是 将 文件 〈 图 片 ) 从 浏览 器 端 复制 到 服务 器 端 指 定 的 文 
件 夹 中 ， 数 据 库 所 存储 的 就 是 文件 〈 图 片 ) 的 相对 地 址 。 当 页 面 显示 图 片 时 ， 实 际 是 分 两 步 : 第 一 步 
是 读 取 数据 表 中 的 地 址 ;第 二 步 是 根据 地 址 找到 并 在 页 面 中 显示 图 片 。 使 用 目录 保存 文件 的 好 处 是 减 
少 了 数据 库 的 容量 和 对 数据 库 的 压力 ， 而 且 图 片 很 容易 被 搜索 引擎 抓 到 ， 从 而 提高 网 站 的 流量 和 人 气 。 

move_uploaded_ file0) 函 数 的 一 般 格式 如 下 : 


bool move_uploaded file ( string filename, string destination ); 


参数 说 明 如 下 。 
filename: 上 传 到 服务 器 中 的 临时 文件 名 。 
destination: 保存 文件 的 实际 路 径 。 


人 注意 这 册 了 RU 为 性 时 文件 各 ， 而 不 是 上 从 文 件 的 原文 件 训 ， 可 以 通过 $_FILES[filename] 
[tmp_name] 来 获取 。 


下 面 看 一 个 实例 ， 程 序 代码 如 下 : 


<label> 请 选择 要 上 传 的 图 片 (文件 小 于 500K): </label> 
<!-- 上 传 文件 的 form 表单 --> 


& 
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<form method="post" action="# enctype="multipart/form-data"> 
<input type="hidden" name="action" value="upload" /> 
<input type="file" name="u_file"/> 
<input type="submit" value=" 上 传 " /> 


</form> 
<! 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 : > 
<?php 
让 判断 是 否 为 上 传动 作 */ 
if(isset($_POST[action])X 
Sfile_path = "./"; // 上 传 文件 存放 路 径 ，./ 为 当前 目录 下 
$profix = array(".jpg",".gif,"jpeg"); // 设 置 允 许 上 传 的 文件 后 缀 类 型 
$f_name = $_FILES[u_file][name]; // 取 得 要 上 传 的 文件 名 
$pro_name=substr($f_name,strrpos($f_name,".")); // 取 得 上 传 文件 的 后 缀 
if(lin_array($pro_name, $profix)){ // 判 断 上 传 文件 的 类 型 是 否 为 允许 类 型 
echo "文件 格式 不 对 "; 
exit(); 
和 
if$_FILES[u_filejlsize] <= OX 1/ 判断 上 传 文件 的 大 小 
echo "文件 上 传 错误 ， 或 文件 大 于 500KB, 请 重新 上 传 "， ”// 如 果 文件 过 大 ， 提 示 错 误 
和 
else{ 
/上 传 文件 函数 "/ 
move_uploaded file($_FILES[u_file][tmp_name],$file_path.$_FILES[u_filelIname]); 
echo "上 传 成 功 ""; 
?> : 


3. 使 用 二 进 制 保存 图 片 

上 传 图 片 的 另 一 种 保存 方式 是 以 二 进 制 的 形式 存储 在 数据 库 中 。 在 计算 机 看 来 ， 再 美的 图 片 、 再 感人 
的 文章 ， 也 不 过 就 是 一 堆 “0111010011…” 的 代码 段 而 已 ， 和 其 他 数据 没有 什么 区 别 ， 存 到 数据 表 中 都 是 
一 样 的。 存储 图 片 不 再 需要 使 用 move_uploads_file0 函 数 ， 在 处 理 页 中 直接 使 用 二 进 制 的 形式 读 取 出 文件 
图 片 ， 并 存 到 数据 表 中 。 本 系统 采用 的 就 是 二 进 制 的 形式 来 保存 图 片 ， 具 体操 作 将 在 实现 过 程 中 给 出 。 

使 用 二 进 制 来 保存 图 片 的 好 处 是 安全 ， 特 别 是 涉及 个 人 隐私 ， 不 易 被 窃取 ， 存储 方便 ， 和 表 中 其 
他 数据 同等 操作 就 可 以 ; 节省 磁盘 空间 ， 由 于 文件 系统 类 型 的 限制 ， 放 到 数据 表 中 的 数据 一 定 比 直接 
放 到 磁盘 中 要 节省 空间 。 

当然 二 进 制 形式 的 图 片 也 有 很 大 弊端 ， 在 传 出 的 过 程 中 ， 如 果 某 个 二 进 制 位 出 现 丢失 或 损坏 ， 那 
么 整 张 图 片 将 无 法 显示 ， 而 是 变 为 一 堆 乱码 ， 而 且 如 果 存 储 的 图 片 过 多 ， 造 成 数据 库 过 于 庞大 ， 将 影 
响 其 他 数据 的 正常 读 取 ， 增 大 服务 器 的 负担 。 所 以 ， 对 于 太 大 的 文件 ， 就 不 适宜 保存 到 数据 库 中 ， 而 
是 应 该 放 到 文件 夹 。 


7.7.3 图 片上 传 的 实现 过 程 


国 图片 上传 使 用 的 数据 表 : tb tpsc 
博客 用 户 登 录 后 ， 单 击 导航 栏 中 的 “图 片 管理 ”/“ 添 加 图 片 ”选项 ， 即 可 进入 添加 图 片 页 面 ， 在 
“图 片 名 称 ” 文 本 框 中 添加 上 传 的 图 片 名 称 ， 在 “上 传 路 径 ”文本 框 中 选择 或 者 单 击 “ 浏 览 ”按钮 选 
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择 自己 喜欢 的 图 片 ， 单 击 “ 提 交 ” 按 钮 ， 以 二 进 制 形式 将 图 片上 传 到 数据 库 中 。 图 片上 传 页 面 的 运行 
结果 如 图 7.25 所 示 。 


CC 5 FM.ONs 


图 7.25 图 片上 传 页 面 的 运行 结果 


图 片上 传 页 是 一 个 上 传 文件 的 表单 ， 主 要 包括 一 个 文本 域 、 一 个 文件 域 和 一 个 “提交 ”按钮 。 部 
分 表单 元 素 的 名 称 及 属性 如 表 7.7 所 示 。 


表 7.7 图 片上 传 页 面 中 的 表单 元 素 


当 用 户 输 入 图 片 名 称 , 并 选择 图 片 路 径 后 , 单 击 “ 提 交 ” 按 钮 , 系统 将 进入 到 上 传 处 理 页 (tptj_ok.php) 
中 进行 处 理 。 在 处 理 页 中 ， 首 先 对 图 片 名 称 进行 处 理 ， 去 掉 特殊 字符 、 空 行 和 空格 ， 然 后 对 上 传 的 文 
件 进 行 类 型 检查 和 文件 大 小 检查 。 最 后 ， 以 二 进 制 的 形式 和 图 片 的 其 他 信息 《如 上 传 用 户 、 上 传 时间 
等 ) 一 起 存 进 数据 表 中 。 关 键 代码 如 下 : 


倒 程 06 ”代码 位 置 ， 光盘 \TM\07\tmlog\tptj_ok.php 


<?php 
session_start(); /| 开启 session 支持 
include "check_login.php"; // 包 含 权 限 检查 文件 
include "Conn/conn.php"; // 包 含 数据 库 链 接 文件 
if($_POST["btn_tj"]==" 提 交 "){ 
[9 $tpmc=htmlspecialchars($tpmc); // 将 图 片 名 称 中 的 特殊 字符 转换 成 HTML 格式 
@ S$tpmc=str_replace("\n","<br>",$tpmce); // 将 图 片 名 称 中 的 回 车 符 以 自动 换行 符 取 代 
Stpmc=str_replace(","&nbsp:",$tpmc); /将 图 片 名 称 中 的 空格 以 “&nbsp;” 取 代 
$author=$_SESSION[username]; 
$scsj=date("Y-m-d"); // 设 置 图 片 的 上 传 时 间 
$profix = array(".jpg",".gif,"jpeg"); // 设 置 允许 上 传 的 文件 后 缀 类 型 
$f_name = $_FILES[file][name]; // 取 得 要 上 传 的 文件 名 
$pro_name=substr($f_name,strrpos($f_name,"."));，// 取 得 上 传 文件 的 后 缀 
/判断 上 传 文件 的 类 型 是 否 为 允许 类 型 */ 
33 if(!in_array(strtolower($pro_name), $profix)\{ 
echo "<script>alert( 文 件 格 式 不 对 ');history.go(-1);</script>"; 
exit(); 


} 
/判断 上 传 文件 的 大 小 ， 如 果 文件 过 大 ， 提 示 错 误 */ 
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if($_FILES[filel[size] > 500000){ 
echo "<script>alert( 文件 上 传 错误 ,请 重新 上 传 ');history.go(-1)</script>"; 


exit(); 
else{ 
$fp=fopen(S$file,"r"); /以 只 读 方式 打开 文件 
$file=addslashes(fread($fp,filesize($file))); /将 文件 中 的 引号 部 分 加 上 反 斜 线 


$query="insert into tb_tpsc (tpmo,file,author,scsj) values ($tpmc',$file',,$author,'$scsj)"; 
$result=mysql_query($query); 
echo "<meta http-equiv=\"refresh\" content=\"1;url=browse_pic.php\"> 图 片上 传 成 功 ， 请 稍 等 …"; 


?> 


< 代码 贴 十 
@ htmlspecialchars(0) 函 数 : 将 特殊 字符 转换 成 HTML 格式 。 
@ str_ replace() 函 数 : 取代 所 有 在 字 囊 中 出 现 的 字 囊 。 语 法 如 下 : 


mixed str_replace ( mixed search, mixed replace, mixed subject , int &count ) 


str_replace() 浮 数 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替换 ,参数 &count 表示 替换 字符 串 执行 的 次 数 。 
@ strtolower: 将 字符 转换 为 小 写字 母 。 


7.7.4 图 片 浏览 的 实现 过 程 


国 图片 上传 使 用 的 数据 表 : tb_tpsc 
无 论 是 注册 用 户 , 还 是 非 注册 用 户 , 只 要 登录 


De 


网 站 ， 就 可 以 无 条 件 地 浏览 所 有 图 片 。 而 删除 图 片 cc Ss no 

除了 管理 员 , 其 他 人 都 无 权 操 作 。 非 注册 用 户 可 以 

通过 首页 中 的 “最 新 图 片 ”进入 图 片 浏览 页 面 , 注 一 一 一 一 -一 一 
册 用 户 先进 入 个 人 管理 界面 ， 单 击 “ 图 片 管理 ”/ i | 


“浏览 图 片 ”菜单 ， 同 样 可 以 进入 图 片 浏览 页 面 。 
注册 用 户 浏览 图 片 页 面 的 运行 结果 如 图 7.26 所 示 。 
本 页 的 实现 代码 和 查看 文章 页 面 略 有 不 同 , 查 


ET 2007.12- 是 20 人 Henglan07.12.14 


看 文章 页 面 中 , 每 条 数据 占 了 一 行 , 而 查看 图 片 则 CA 
采用 的 是 分 栏 显示 ， 以 每 行 两 张 图 片 的 格式 输出 ， 图 7.26 浏览 图 片 页 面 的 运行 结果 


每 页 显示 4 张 图 片 。 通过 单 击 小 图 片 , 可 以 查看 图 
片 原 效 果 图 。 程 序 关键 代码 如 下 : 
全 性 07 ”代码 位 置 ， 光盘 \TM\07\tmlog\browse_pic.php 


<?php 
session_start(); /开启 session 支持 
include "Conn/conn.php"; // 包 含 数据 库 链 接 文 件 
include "check_login.php"; // 包 含 权限 检查 文件 
?> 


<!- 首 部 导航 栏 -> 
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<tr align="left" colspan="2" > 

<td width="390" height="25" colspan="3" valign="top" bgcolor="#EFF7DE"> <span class="tableBorder_ 
LTR"> 浏 览 图片 </span></td></tr> 

<tr> 

<td height="192" align="center valign="top" > 

<?php 

全 /分 页 显示 

$query="select * from tb_tpsc where scsj order by id desc limit $offset, $page_size"; 

"返回 结果 集 */ 

$result=mysql_query ($query); 
?> 
<table width="496" border="1" align="center" cellpadding="3" cellspacing="1" bordercolor="#D6D7D6"> 

<tr> 
<?php 
/设置 一 个 变量 ， 判 断 当前 数据 是 奇数 输出 还 是 偶数 输出 
$i=1; 
while($info=mysql_fetch_array($result) 


{ 
A* 如 果 当前 为 偶数 输出 ， 记 录 输 出 完毕 后 结束 <tr> 标 签 */ 
if($i%2==0X{ 
> 
<td width="500"><table width="245" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td colspan="2"><div align="center"> 
<!-- 显 示 图 片 页 image.php 一 > 
<a href="image.php?recid=<?php echo $info[id]; ?>" target="_blank"><img src= 
"image.php?recid=<?php echo S$info[id];?>" width="150" height="100"></a></div></td></tr> 
<tr> 
<td width="109" height="25" align="left">&nbsp; 图 片 名 称 :<?php echo S$infoftpmc];?> 
</td> 
<td width="128"> 上 传 时 间 :<?php echo S$info[scsj];?></td></tr> 
<tr> 
<td colspan="2" height="25"> 
<?php 
让 如 果 登 录用 户 为 管理 员 ， 显 示人 删除 操作 */ 
if ($_SESSIONIfigj==1){ 
?> 
<a href="remove.php?pic_id=<?php echo Sinfolid]?>"><img src="images/ 
A_delete.gif' width="52" height="16" alt=" 删 除 图 片 " onClick="return pic_chk();"></a> 


<?php 
} 
?> 
</td> 
</tr> 
</table></td> 
</tr> 
<?php 


} 
否则， 开始 新 的 一 行 */ 
else 


{ 


@ 
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?> 
<td width="500" ><table width="236" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td colspan="2"><div align="center><a href="image.php?recid=<?php echo S$infofid]; ?>" target="_blank"> 
<img src="image.php?recid=<?php echo S$info[id];?>" width="150" height="100"></a></div></td> 
</tr> 
<tr> 
<td width="110" height="25">&nbsp; 图 片 名 称 :<?php echo $info[tpmc];?></td> 
<td width="126"> 上 传 时 间 :&nbsp;<?php echo S$info[scsj];?></td> 
</tr> 
<tr> 
<td colspan="2" height="25"> 
<?php 
if($_SESSION[fig]==1){ 
?> 
<a href="remove.php?pic id=<?php echo Sinfofid]?>"><img src="images/ 
A_delete.gif" width="52" height="16" alt=" 删 除 图 片 " onClick="retum pic_chk();"></a> 


<?php 
} 
?> 
</td> 
</tr> 
</table></td> 
<?php 
} 
/变量 $i 自 加 1，while 循环 结束 */ 
Si++; 
} 
?> 
</tr> 


通过 上 述 代码 可 以 看 到 ， 预 览 图 片 和 显示 图 片 实际 调用 的 都 是 同一 页 面 : image.php。image.php 页 
就 是 用 来 显示 图 片 的 实际 效果 的 。 实 现代 码 如 下 : 
倒 程 08 ”代码 位 置 ， 光盘 \TM\07\tmlog\image.php 


<?php 
include "Conn/conn.php"; // 包 含 数据 库 链 接 文件 
$query="select id,tpmcfile,authorscsj from tb_tpsc where id=".$recid; /根据 id 生成 查询 语句 
$result=mysql_query($query); /执行 查询 语句 
if(l$result) die("error: mysql query"); // 判 断 是 否 有 返回 结果 
$num=mysql_num_rows($result); 1/ 计算 结果 集 个 数 
if($num<1) die("error: no this recorder"); 
$data = mysql_result($result,0,"file"); /| 返回 图 片 数据 
echo $data; /| 输 出 图 片 
?> 


4 注意 mysql_result() 函 数 不 能 和 其 他 处 理 结果 集 函 数 混 合 调用 。 
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7.7.5 删除 图 片 的 实现 过 程 


国 删除 图 片 使 用 的 数据 表 : tb tpsc 

删除 图 片 是 管理 员 才 有 的 权限 ， 在 图 片 浏览 的 实现 代码 中 ， 已 经 给 出 如 何 判断 登录 用 户 是 否 为 管 
理 员 。 当 管理 员 单 击 “ 删 除 ” 超 链接 时 ， 处 理 页 (remove.php) 会 根据 传 过 来 的 id， 删除 对 应 的 数据 
表 中 的 数据 。 实 现代 码 如 下 : 

倒 程 09 ”代码 位 置 : 光盘 \TM\07\tmlog\remove.php 


<?php 
session_start(); /开启 session 支持 
include "check_login.php"; // 包 含 权限 检查 文件 
include "Conn/conn.php"; // 包 含 数 据 库 链接 文件 
if($_session[fig] != "1" // 判 断 是 否 为 管理 员 
echo "<script>alert(' 您 没有 删除 权限 ');history.go(-1);</script>"; 
} 
$sql="delete from tb_tpsc where id=".$pic_id; /| 生成 删除 语句 
$result=mysql_query($sql); // 执 行 删 除 语句 
if($result){ // 根 据 $result 返回 结果 
echo "<script>alert(' 图 片 删除 成 功 !);location='browse_pic.php';</script>"; 
} 
else{ 
echo "<script>alert(' 图 片 删 除 操作 失败 !");history.go(-1);</script>"; 
3 
?> 


7.7.6 单元 测试 


在 进行 上 传 模块 的 测试 时 发 现 : 当 上 传 比较 大 的 图 片 时 ， 文 件 大 小 的 验证 部 分 并 没有 生效 ， 而 是 
直接 跳 过 。 在 处 理 页 中 产生 错误 ， 并 且 显 示 页 面 不 显示 图 片 。 运 行 结果 如 图 7.27 所 示 。 


村 


mlog\tptj_ok. php cn 1 


日 
| B007. 
df ET 2007 12- tS Ek Leer 2007.12.20 
xm xm 
机 :1 记录 :5 条 下 一 贡 尾 页 


图 7.27 上 传 图 片 错误 信息 及 显示 结果 
查看 原文 件 大 小 验证 的 代码 如 下 : 


/判断 上 传 文件 的 大 小 ， 如 果 文件 过 大 ， 提 示 错 误 */ 

if($_FILESIflellsize] > 500000){ 
echo "<script>alert( 文 件 上 传 错误 ， 请 重新 上 传 ');history.go(-1)</script>"; 
exit(); 

上 


@ 
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经 反复 测试 后 发 现 : 如 果 上 传 图 片 大 于 500KB 而 小 于 2MB 时 ， 原 代码 生效 。 如 果 图 片 大 于 2MB， 
则 代码 被 跳 过 ， 这 时 输出 $_FILES[file][size] 的 值 为 0。 原来， 在 配置 文件 phpini 中 ， 
upload_mix_filesize=2MB， 也 就 是 说 ， 上 传 文件 的 大 小 超过 了 系统 设置 的 值 ， 那 么 $_FILES[file][size] 
返回 0， 在 判断 表达 式 中 返回 true， 代 码 自然 被 跳 过 。 

改写 验证 代码 ， 更 改 后 的 代码 如 下 : 

倒 程 10 ”代码 位 置 ， 光 盘 \TM\07\tmlog\tptj_ok.php 

if(($_FILESIfile][size] <= 0) or ($_FILEST[file][size] > 500000)){ 


echo "<script>alert(' 文 件 上 传 错误 ， 或 文件 大 于 500KB， 请 重新 上 传 ');history.go(-1)</script>"; 
exit(); 


} 
进行 二 次 测试 ， 问 题 没有 再 出 现 ， 系 统 正常 运行 。 


7.8 ”朋友 圈 模 块 设 计 


7.8.1 朋友 圈 模 块 概述 


本 系统 的 朋友 圈 模 块 的 主要 功能 是 添加 、 查 询 、 删 除 好 友 ， 添 加 的 好 友 除 了 该 用 户 以 外 ， 包 括 管 
理 员 在 内 的 所 有 外 人 都 不 可 以 查看 , 以 保证 其 个 人 隐私 
不 被 外 泄 。 用 户 被 删除 时 ,该 用 户 现 有 的 朋友 圈 也 一 并 
被 删除 。 朋 友 圈 模块 框架 图 如 图 7.28 所 示 。 


7.8.2 ”朋友 圈 模 块 技术 分 析 


在 查询 好 友 的 功能 中 , 使 用 到 了 模糊 查询 语句 , 用 
于 模糊 查找 好 友 列 表 。 模 糊 查 询 语句 使 用 的 是 like 运算 
符 。 在 PHP 中 ， 带 有 like 运算 符 的 查询 语句 的 常用 格 
式 有 以 下 两 种 : 
(1) 使 用 通配符 “%” 的 where 子 句 图 7.28 朋友 圈 模 块 框架 图 
通配符 “%” 表 示 0 个 或 多 个 任意 长 度 和 类 型 的 字 
包括 中 文 汉字 。 
示例 1: 表示 查找 所 有 内 容 包 含 “ 好 ” 字 的 文章 。 
select * from tb_file where content like '% 好 %'; 
示例 2: 查找 所 有 包含 “好 ” 字 或 “高 ” 字 的 文章 ， 这 时 可 以 配合 or 运算 符 来 使 用 。 代 码 如 下 : 


select * from tb_file where content like '% 好 %' or content like '% 高 %'; 


添加 好 友 


1 
好 友 列 表 


| 


(2) 使 用 通配符 “_” 的 where 子 句 
通配符 “_ ”表示 匹配 任意 的 单个 字符 。 
示例 1: 查找 用 户 名 只 包含 5 个 字符 ， 其 中 后 4 个 字符 为 soft 的 用 户 。 代 码 如 下 : 
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select* from tb_user where regname like "soft ; 
示例 2: 查找 所 有 以 + 开头， 并 且 以 + 结尾 的 、 中 间 包 含 3 个 字符 的 用 户 。 代 码 如 下 : 
select* from tb_user where regname like t__t; 


查找 的 结果 为 : tsoft。 


人 
据 ， 或 返回 的 数据 不 匹配 。 所 以 在 安装 MySQL 时 ， 要 保持 和 系统 编码 的 统一 。 常 用 的 编码 格式 有 
gb2312、ISO-8859-1、ut 和 gbk 


7.8.3 ”查询 好 友 的 实现 过 程 


国 ”查询 好 友 使 用 的 数据 表 : tb_friend 

当 用 户 要 查询 好 友 时 ， 单 击 “ 朋 友 圈 管理 ”/“ 查 询 朋 友信 息 ”， 显 示 查 询 页 面 。 查 询 可 以 分 为 姓 
名 查询 和 编号 查询 ， 均 为 模糊 查询 。 当 用 户 输 入 要 查找 的 关键 字 后 ， 单 击 “ 检 索 ” 按 钮 ， 或 按 Enter 
键 ， 系 统 跳 到 处 理 页 进行 处 理 。 查 询 好 友 页 面 的 运行 结果 如 图 7.29 所 示 。 
| 
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图 7.29 查询 好 友 页 面 的 运行 结果 


查询 页 包含 一 个 查询 表单 ， 包 括 查询 条 件 和 查询 关键 字 两 部 分 表单 元 素 。 主 要 表单 元 素 如 表 7.8 


表 7.8 查询 页 表单 的 主要 元 素 属性 


method="post" action="guery_friend.php" onSubmit="return checkO:” 查询 好 友 表 单 
<option value="name" selected> 姓 名 </option> 2 
本 select l 查询 条 件 选择 
<option value="id"> 编 号 </option> 
sel key | text | id="sel key" size="30" | 查询 关键 字 
submit submit type="submit" name="submit" value=" 检 索 " “检索 ”按钮 


当 处 理 页 接收 到 查询 条 件 及 查询 关键 字 后 ， 生 成 模糊 查询 语句 ， 执 行 SQL 语句 并 返回 查询 结果 。 
如 果 没 有 输入 关键 字 ， 则 弹出 提示 框 ; 如 果 没 有 查找 到 任何 结果 ， 则 输出 “Sorry! 没 有 您 要 找 的 朋友 !”。 
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处 理 页 的 关键 代码 如 下 : 
倒 性 11 ”代码 位 置 光盘 \TM\07\tmlog\query_friend.php 
<?php 


session_start(); /开启 session 支持 
include "Conn/conn.php"; /包含 数据 库 链接 文件 
include "check_login.php"; /包含 权限 检查 文件 
?> 


<script language="javascript"> 
/*JavaScript 脚本 ， 验 证 输入 信息 ， 如 果 为 空 ， 则 弹出 提示 框 */ 


function check(formX{ 
if (document.myform.sel_key.value=="" 
alert(" 请 输入 查询 条 件 !"); 
myform.sel_keyfocus(); 
return false; 
} 
} 
</script> 
<?php 
if ($_POST[sel_key]!=")}{ 
$tj=$_POSTIsel 1]; // 接 收 查询 字段 
$key=$_POST[sel_key]; // 接 收 查询 关键 字 
$sql=mysql_query("select * from tb_friend where $t] like '%$key%' and username='$_SESSION 
[usemame]"); 
$result=mysql_fetch_array($sql); /执行 查询 语句 


if($result==false){ 
echo ("[<font color=red>Sorry! 没 有 您 要 找 的 朋友 !</font>]"); 
Jelse{ 


?> 
人 <!-- 显 示 记 录 表 单 -> 
<?php }?> 


7.9 开发 技巧 与 难点 分 析 


想象 一 下 : 动态 的 下 拉 菜单 ， 半 透明 的 背景 效果 ， 鼠 标 划 过 时 的 色彩 交替 ， 给 人 一 种 亦 真 亦 幻 的 
视觉 冲击 和 美的 感受 ， 这 样 的 网 站 是 不 是 具有 更 大 的 吸引 力 ? 本 系统 个 人 管理 页 面 中 的 管理 菜单 实现 
的 就 是 这 种 效果 。 当 用 户 登录 到 博客 网 站 后 ， 将 鼠标 移动 到 “朋友 圈 管 理 ” 等 导航 链接 上 时 ， 将 在 下 
方 显示 出 半 透 明 的 下 拉 式 菜 单 ， 透 过 此 下 拉 菜 单 仍 可 以 看 到 页 面 上 的 内 容 ， 运 行 效果 如 图 7.30 所 示 。 
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图 7.30 动态 半 透 明 效果 的 下 拉 菜 单 


PHP 项 目 开发 全 程 实录 (第 3 版 ) 


实现 半 透 明 背 景 的 下 拉 菜 单 ， 首 先 需要 在 页 面 中 实现 下 拉 菜 单 ， 然 后 再 通过 设置 下 拉 菜 单 的 CSS 
样式 实现 半 透 明 效 果 。 实 现下 拉 菜 单 的 半 透 明 效 果 可 以 应 用 CSS 样式 的 透明 效果 滤 镜 alpha 实现 。alpha 
属性 是 把 一 个 目标 元 素 与 背景 混合 。 这 种 “与 背景 混合 ”， 通 俗 地 说 ， 是 一 个 元 素 的 透明 度 。 透 明 效 果 
滤 镜 alpha 的 语法 如 下 : 


{filter:alpha(opacity=opacity,finishopacity=finishopacity,style=style, startx=startx, starty=starty,finishx=finishx,finishy 
=finishy)} 
滤 镜 alpha 的 各 个 参数 说 明 如 表 7.9 所 示 。 

表 7.9 alpha 滤 镜 的 各 个 参数 及 说 明 


属 性 说 明 

| 代表 透明 度 水 准 。 默 认 的 范围 是 0 一 100， 其 实 是 百分比 的 形式 ， 也 就 是 0 代表 完全 透明 ，100 代表 完 
py 全 不 透明 

finishopacity | 可 选 ， 如 果 想 要 设置 渐变 的 透明 效果 ， 可 以 使 用 该 参数 指定 结束 时 的 透明 度 。 范 围 也 是 0~-100 


style 指定 透明 区 域 的 形状 特征 ， 其 中 0 代表 统一 形状 、1 代表 线形 、2 代表 放射 状 、3 代表 长 方形 
startx 代表 渐变 透明 效果 的 开始 X 坐标 
starty 代表 渐变 透明 效果 的 开始 Y 坐标 
finishx 代表 渐变 透明 效果 的 结束 X 坐标 
finishy 代表 渐变 透明 效果 的 结束 Y 坐标 


在 本 实例 中 的 menuskin 类 中 ， 设 置 了 alpha 滤 镜 的 属性 。 代 码 如 下 : 
倒 程 12 代码 位 置 ， 光盘 \TM\07\tmlog\css\style.css 


.menuskin { 

BORDER: #666666 1px solid; // 层 边框 样式 

VISIBILITY: hidden; // 层 可 见 为 隐藏 

FONT: 12px Verdana; /| 层 中 的 字体 样式 和 大 小 

POSITION: absolute; /定位 方式 

background-image:url("../images/item_out.gif"); 1/ 背景 图 片 

background-repeat : repeat-y; /图 片 是 否 可 重复 

filter: alpha(Opacity=85); // 设 置 alpha 的 opacity 属性 等 于 85， 即 透明 度 为 85% 
} 


在 显示 页 面 中 显示 菜单 项 的 表单 引入 这 个 menuskin 类 ， 当 鼠标 划 过 特定 的 文字 时 ， 将 会 发 现 自动 
下 拉 菜 单 的 效果 为 半 透 明 的 样式 。 页 面 关键 代码 如 下 : 

倒 程 13 ”代码 位 置 光盘 \TM\07\tmlog\file.php 

<!- 弹 出 层 设置 -> 

<div class=menuskin id=popmenu 
<!-- 鼠 标 进入 该 区 域 时 调用 的 js 方法 --> 
onmouseover="clearhidemenu();highlightmenu(event,on )” 
<!-- 鼠 标 离开 该 区 域 时 调用 的 js 方法 --> 
onmouseout="highlightmenu(event, of );dynamichide(event)” 
<!-- 设 置 层叠 顺序 和 定位 方式 -> 
style="Z-index:100;position:absolute;"> 

</div> 


@ 
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<!-- 半 透明 的 动态 下 拉 菜单 -> 
<TABLE style="WIDTH: 580px" VERTICAL-ALIGN: text-top; cellSpacing=0 cellPadding=0 align="center"> 
<TR align="center" valign="middle"> 

<!-- 显 示 登 录用 户 --> 

<TD style="WIDTH: 100px; COLOR: red;"> 欢 迎 您 :&nbsp;<?php echo 
$_SESSION[usemame]; ?>&nbsp;&nbsp;</TD> 

<!-- 显 示 首页 超 链接 一 > 

<TD style="WIDTH: 80px; COLOR: red;"><SPAN style="FONT-SIZE: 9pt; COLOR: #cc0033"> 
</SPAN><a href="index.php"> 博 客 首页 </a></TD> 

<!-- 显 示 文 章 管理 下 拉 菜 单 -> 

<TD style="WIDTH: 80px; COLOR: red;"><a onmouseover=showmenu(event,productmenu) 
onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand" > 文章 管理 </a></TD> 

<!-- 显 示 图 片 管理 下 拉 菜 单 -> 
<TD style="WIDTH: 80px; COLOR: red;"><a onmouseover=showmenu(event,Honourmenu) 
‘onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand"> 图 片 管理 </a></TD> 

<!-- 显 示 朋友 团 管 理 下 拉 菜 单 -> 

<TD style="WIDTH: 90px; COLOR: red;"><a onmouseover=showmenu(event,myfriend) 
onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand" > 朋友 图 管理 </a></TD> 
<?php 

"如果 登录 用 户 为 管理 员 */ 

if($_SESSIONI[fig]==1)X{ 
?> 

<!- 一 一 一 一 一 一 一 一 一 一 显示 管理 员 管 理 下 拉 菜 单一 一 一 一 一 一 一 一 一 一 一 一 -- > 

<TD style="WIDTH: 80px; COLOR: red;"><a onmouseover=showmenu(event,myuser) 
onmouseout=delayhidemenu() class='navlink' style="CURSOR:hand" > 管理 员 管理 </a></TD> 


<?php 
} 
?> 
< 一 一 一 一 一 -一 -显示 退出 超 链接 一 一 
<TD style="WIDTH: 80px; COLOR: red;"><a href="safe.php"> 退 出 登录 </a></TD> 
</TR> 
</TABLE> 


通过 代码 可 以 发 现 ， 在 每 个 超 链接 标签 <a> 中 ， 都 有 3 个 参数 ， 分 别 介 绍 如 下 。 

onMouseOver: 鼠标 进入 该 区 域 时 的 效果 。 本 页 面 调用 了 js 方法 showmenu()。 

onMouseOut: 鼠标 移出 该 区 域 时 的 效果 。 本 页 面 调用 js 方法 delayhidemenu()。 

style: 标签 样式 ， 本 页 面 样式 为 “CURSOR:hand;”， 显示 一 个 手 的 形状 。 

调用 的 js 方法 就 是 实现 下 拉 菜 单 的 JavaScript 代码 : 显示 和 隐藏 菜单 。 由 于 这 段 代码 是 通用 型 代 
码 ， 所 以 单独 放 到 一 个 js 文件 中 ， 方 便 其 他 页 调用 ，js 文件 名 为 menu.js。 调 用 时 在 页 面 中 加 入 如 下 代 
码 即 可 。 


<script src="JS/menu.js"></script> 


menujs 文件 中 显示 、 隐 藏 菜单 和 变换 背景 等 关键 函数 的 代码 如 下 : 
倒 程 14 ”代码 位 置 ， 光盘 \TM\07\tmlog\js\menu.js 
/浏览 器 版 本 号 检测 中 


var IE4=document.all&&navigator.userAgent.indexOf("Opera" 


301 
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var netscape6=document.getElementByld&&ldocument.all 
var netscape4=document.layers 
/显示 下 拉 菜单 函数 
function showmenu(e,vmenu,mod)j{ 
让 判断 浏览 器 版 本 */ 
if (Idocument.all&&!document.getElementByld&&!document.layers) 
return 
/菜单 内 容 */ 
which=vmenu 
广 调用 clearhidemenu() 函 数 中 
clearhidemenu() 
/* 调 用 IE_clearshadow() 函 数 */ 
IE_clearshadow() 
根据 不 同 的 浏览 器 ， 声 明 弹出 式 菜单 *%/ 
menuobj=IE4? document.all.popmenu : netscape6? document.getElementByld("popmenu") : netscape4? 
document.popmenu : ™" 
menuobj.thestyle=(IE4||netscape6)? menuobj.style : menuobj 
if(IE4|lInetscape6) 
/如 果 是 IE 4 或 Netscape 6 浏览 器 ， 就 调用 innerHTML 方法 输出 下 拉 菜 单 */ 
menuobj.innerHTML=which 
else{ 
/否则 使 用 write 方法 输出 下 拉 菜 单 */ 
menuobj.document.write('<layer name="other bgColor="#E6E6E6" width="165" 
onmouseover="clearhidemenu()" onmouseout="hidemenu()">'+which+'</layer>') 
人 关闭 数据 流 */ 


menuobj.document.close() 


} 
人 * 调 用 不 同 的 方法 属性 ， 来 设置 不 同 浏览 器 的 宽 、 高 及 内 部 表格 的 大 小 、 位 置 */ 
menuobj.contentwidth=(IE4||netscape6)? menuobj.offsetWidth : menuobj.document.other.document.width 
menuobj.contentheight=(IE4|lnetscape6)? menuobj.offsetHeight : menuobj.document.other.document. height 
eventX=IE4? event.clientX : netscape6? e.clientX : e.x 
eventY=IE4? event.clientY : netscape6? e.clientY : e.y 
var rightedge=IE4? document.body.clientWidth-eventX : window.innerWidth-eventX 
varbottomedge=IE4? document.body.clientHeight-eventY : window.innerHeight-eventY 
if (rightedge<menuobj.contentwidth) 
menuobj.thestyle.left=IE4? document.body.scrollLeft+reventX-menuobj.contentwidth+menuOffX : 
netscape6? window.pageXOffset+eventX-menuobj.contentwidth : eventX-menuobj.contentwidth 
else 
menuobjthestyle.left=IE4? IE_x(eventsrcElement)+tmenuOfX : netscape6? window.pageXOffset+eventX : 
eventX 
if (bottomedge<menuobj.contentheight&&mod!=0) 
menuobj.thestyle.top=IE4? document.body.scrollTop+eventY-menuobj.contentheight-event. offsetY+ 
menuOffY-23 : netscape6? window.pageYOffset+eventY-menuobj.contentheight-10 : eventY-menuobj.contentheight 
else 
menuobj.thestyle.top=IE4? IE_y(event.srcElement)+menuOffY : netscape6? window.pageYOffset+ 
eventY+10 : eventY 
让 设置 标签 样式 为 可 见 */ 
menuobj.thestyle.visibility="visible” 
/* 调 用 IE_dropshadow() 方 法 */ 


IE_dropshadow(menuobj,"#999999",3) 
@ 


retumn false 
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/隐藏 下 拉 菜 单 函数 ”/ 
function hidemenu(X{ 
if (window.menuobj) 
menuobj.thestyle.visibility=(IE4|Inetscape6)? "hidden" : "hide”" 
IE_clearshadow() 
} 
function dynamichide(eX{ 
if (IE4&&!mMenuobj.contains(e.toElement)) 
hidemenu() 
else if (netscape6&&e.currentTarget!= e.relatedTarget&& !contains_netscape6(e.currentTarget, e.relatedTarget)) 
hidemenu() 
} 
/延迟 隐藏 下 拉 菜单 函数 "/ 


function delayhidemenu(X{ 
if (lIE4|lInetscape6|Inetscape4) 
delayhide=setTimeout("hidemenu()",500) 


} 
/停止 隐藏 菜单 函数 "/ 
function clearhidemenu(){ 
if (window.delayhide) 
clearTimeout(delayhide) 
} 


/* 设 置 菜 单 背景 */ 

/* 鼠 标 移入 该 区 域 时 显示 的 背景 */ 

function overbg(tdbg){ 
tdbg.style.background='url( images/item_over.gif)' 
tdbg.style.border=' #9CA6C6 1px solid' 


} 

/* 鼠 标 移出 该 区 域 时 显示 的 背景 */ 

function outbg(tdbgX{ 
tdbg.style.background='url(images/item_out.gif) 
tdbg.style.border=" 


3 

/下拉 菜 单 内 容 */ 

/* 文 章 管 理 下 拉 菜 单 */ 

var productmenu='<table width=90><tr><td id=fileadd onMouseOver=overbg(fileadd) onMouseOut=outbg (fleadd)> 
<a href=file.php> 添 加 博客 文章 </a></td></tr>\ 

<tr><td id=query onMouseOver=overbg(query) onMouseOut=outbg(query)><a href=query.php> 查 询 博 客 文章 
</a></td></tr>\ 

<tr><td id=myfiles onMouseOver=overbg(myfiles) onMouseOut=outbg(myfiles)><a href=myfiles.php> 我 的 文章 
</a></td></tr></table>" 

/* 图 片 管理 下 拉 菜 单 */ 

var Honourmenu='<table width=90><tr><td id=picadd onMouseOver=overbg(picadd) onMouseOut= outbg 
(picadd)><a href=add_pic.php> 添 加 图 片 </a></td></tr>\ 

<tr><td id=browse onMouseOver=overbg(browse) onMouseOut=outbg(browse)><a href=browse_pic.php> 浏 览 
图 片 </a></td></tr>\ 

<tr><td id=querypic onMouseOver=overbg(querypic) onMouseOut=outbg(querypic)><a href=query_pic.php> 查 


询 图 片 </a></td></tr></table>' 
® 
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/* 朋 友 图 管理 下 拉 菜 单 */ 

var myfriend='<table width=90><tr><td id=friendadd onMouseOver=overbg(friendadd) onMouseOut=outbg (friendadd)> 
<a href=friend.php> 添 加 到 朋友 圈 </a></td></tr>\ 

<tr><td id=browse fi onMouseOver=overbg(browse 而) onMouseOut=outbg(browse_fri)><a href=browse_ fri.php> 
浏览 我 的 朋友 </a></td></tr>\ 

<tr><td id=cxfriend onMouseOver=overbg(cxfriend) onMouseOut=outbg(cxfriend)><a href=query_friend.php> 查 
询 朋 友信 息 </a></td></tr></table>' 

/管理 员 管 理 下 拉 菜 单 */ 

var myuser='<table width=90><tr><td id=queryuser onMouseOver=overbg(queryuser) onMouseOut=outbg(queryuser)> 
<a href=queryuser.php> 查 询 用 户 信 息 </a></td></tr>\ 

<tr><td id=browseuser onMouseOver=overbg(browseuser) onMouseOut=outbg(browseuser)><a href=browseuser. 
php> 浏 览 用户 信 息 </a></td></tr>\ 

<tr><td id=managepub onMouseOver=overbg(managepub) onMouseOut=outbg(managepub)><a href=managepub. 
php> 公告 管理 </a></td></tr>\ 

</table>" 


7.10 登录 验证 码 技术 专题 


当今 网 络 安全 越 来 越 受到 人 们 的 重视 , 开发 一 个 带 验证 码 的 用 户 登录 模块 可 以 提高 网 站 的 安全 性 。 
其 中 ， 验 证 码 是 随机 生成 的 ， 可 以 是 字母 、 数 字 或 汉字 ， 也 可 以 是 图 片 。 本 节 技术 专题 主要 就 是 学 习 
一 下 验证 码 的 几 种 实现 方法 及 技巧 。 


7.10.1 简单 的 数字 验证 


先 来 看 一 个 简单 的 数字 验证 码 : 生成 一 组 4 位 的 随机 数字 ， 
每 刷新 一 次 ， 显 示 的 结果 都 不 相同 。 如 果 验 证 码 输 入 错误 ， 将 弹 


简单 的 数字 验证 要 


am 1529 


出 提示 窗口 。 数 字 验 证 码 运行 结果 如 图 7.31 所 示 。 | | 
简单 的 数字 验证 码 在 开发 过 程 中 主要 应 用 以 下 两 个 函数 。 LN ware 
(1) mt_rand() 函 数 


如 下 : 


int mt_rand([int min], [int max]); 


例如 ， 从 52 一 79 之 间 取 一 个 随机 数值 。 代 码 如 下 : 


mt_rand(52,79) 


(2) intval0 函 数 
intval(0) 函 数 主要 用 于 将 变量 转 成 整数 类 型 。 语 法 如 下 : 


int intval(mixed var int [base]); 


intval() 函 数 可 将 变量 转 成 整数 类 型 。 可 省 略 的 参数 base 是 转换 的 基数 ， 默 认 值 为 10。 转 换 的 变量 
var 可 以 为 数组 或 类 之 外 的 任何 类 型 变量 。 


人 @ 
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7.10.2 ”数字 图 形 验证 码 


数字 图 形 验 证 码 相 对 于 数字 验证 码 而 言 , 要 稍 复杂 一 些 。 数 字 图 加 
形 验证 码 主要 应 用 mt_rand0) 函 数 来 初始 化 一 组 4 位 的 随机 数 ， 然 后 i = 
利用 for 循环 语句 将 随机 生成 的 4 位 随机 验证 码 利用 数字 图 形 进行 输 画 到 
出 。 图 形 验证 码 的 运行 结果 如 图 7.32 所 示 。 图 732 数字 图 形 验 证 码 
实现 数字 图 形 验证 码 的 关键 代码 如 下 : 


<!-js 脚本 ， 判 断 文本 框 中 输入 的 验证 码 是 否 匹配 -> 
<script language="javascript"> 
function check(form){ 
if(form.txt_yan.value==""){ 
alert(" 请 输入 验证 码 ");form.txt_yan.focus();return false; 


形 皮 证 吗 


} 
if(form.txt_yan.value!=form.txt_hyan.valueX{ 
alert(" 对 不 起 ， 您 输入 的 验证 码 不 正确 ");form.txt_yan.focus();return false; 
} 
} 


</script> 


<!-- 验 证 表单 -> 
<form name="form" method="post" action="" onSubmit="return check(form);"> 
<!-- 验 证 码 文本 框 -> 
<input type="text" name="txt_yan"> 
<!-- 生 成 随机 数字 图 形 验证 码 --> 
<?php 
<!- 生 成 随机 4 位 数 --> 

$num=intval(mt_rand(1000,9999)); 
<!-- 使 用 for 循环 ， 输 出 随机 4 位 数 的 图 片 格式 --> 

for($i=0;$i<4;$i++)f 

echo "<img src=images/checkcode/".substr(strval($num),$i,1).".gif>"; 


j?> 
<!-- 将 随机 数 存 入 隐藏 域 -> 
<input type="hidden" name="txt_hyan" id="txt_hyan" value="<?php echo $num;?>"> 
<br> <br> 
<input type="submit" name="Submit" value=" 验 证 "> &nbsp; <input type="reset" name="Submit2" value=" 重 置 "> 
<!-- 处 理 代码 段 -> 
<?php 
/* 判 断 验证 码 录入 框 是 否 为 空 */ 

if($_POSTI[txt_yan]l=™"){ 
/* 当 输入 的 验证 码 和 隐藏 域 中 的 验证 码 相等 时 ， 输 出 欢迎 信息 */ 

if($_POSTI[txt_yan] == $_POST[txt_hyan]){f 
echo "您 输入 的 验证 码 通过 ， 感 谢 您 的 加 盟 .…"; 


?> 
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7.10.3 ”汉字 图 形 验证 码 


汉字 图 形 验证 码 是 目前 网 络 上 比较 流行 的 验证 码 方式 ， 这 种 验证 码 的 实现 有 一 定 的 难度 。 首 先 将 
图 片 所 对 应 的 汉字 依次 存储 到 数组 中 。 然 后 利用 Rand(O) 函 数 生成 一 个 小 于 10 的 数 ， 用 于 得 到 随机 字符 
的 位 置 ， 进 而 得 出 随机 字符 ， 并 设置 要 显示 的 随机 图 片 。 由 于 产生 的 随机 字符 串 是 4 位 的 ， 所 以 需要 
使 用 for 循环 语句 ， 根 据 数组 得 出 相应 图 片 的 路 径 ， 再 将 图 片 转换 成 数组 中 的 文字 ， 最 后 输出 到 浏览 
器 中 。 
以 汉字 图 形 化 显示 验证 码 的 完整 代码 如 下 : 
<?php 
$str=array(" 大 "更 "" 创 "" 天 "," 科 "" 客 "" 情 "" 技 "" 立 "新 "); 
$word=strlen($str); 
for($i=0;$i<4;$i++){ 
$num=rand(0, $word*2-1); 
$img=$img."<img src='" images/checkcode/".$num.".gif width='16' height='16'>"; // 显 示 随 机 图 片 
$pic=$pic.$str[$num]; /将 图 片 转换 成 数组 中 的 文字 


7.11 本 章 总 结 


本 章 的 博客 管理 系统 首先 介绍 了 博客 的 基本 概念 、 发 展 前 景 、 影 响 范围 及 博客 网 的 功能 分 类 ， 使 
读者 对 当今 主流 博客 有 了 一 个 大 致 的 认识 。 其 次 ， 实 现 了 一 个 博客 系统 ， 其 中 包含 所 有 基本 功能 的 项 
目 开 发 ， 使 读者 对 如 何 开 发 一 个 博客 网 站 有 了 一 个 初步 的 了 解 。 最 后 ， 希 望 读 者 通过 自己 的 努力 ， 来 
逐步 完善 和 加 强 这 个 博客 网 的 实用 功能 ， 最 终 达到 一 个 令 自己 满意 的 作品 。 


全 各 < 


EE 时 
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在 全 球 知识 经 济 和 信息 化 高 速 发 展 的 今天 ， 网 络 化 是 企业 发 展 的 
趋势 ，21 世纪 的 人 更 习惯 在 网 站 上 听 音 乐 、 看 电影 ， 所 以 企业 要 在 同 
领域 中 得 到 突飞猛进 的 发 展 ， 就 必须 借助 网 络 。 

当今 社会 进入 了 一 个 信息 快速 发 展 的 社会 ， 在 网 络 世 界 浏 览 信息 
的 同时 也 可 以 听 听 歌曲 ， 这 样 既 愉 悦 了 身心 ， 又 得 到 了 最 新 的 市 场 动 
态 ， 由 此 网 络 上 也 出 现 了 很 多 的 影视 网 站 ， 都 很 受到 欢迎 。 未 来 视听 
生活 的 新 空间 ， 也 必然 在 宽带 互联 网 上 开启 。VOD 的 概念 已 经 被 越 来 
越 多 的 人 所 接受 ， 逐 渐 成 为 网 络 发 展 的 必然 趋势 之 一 。 通 过 阅读 本 章 ， 
可 以 学 习 到 : 

MW ”online 影视 365 类 网 站 的 整体 设计 思路 及 实现 方法 
应 用 <embed> 和 <object> 播 放 影音 文件 
关闭 子 窗口 时 自动 刷新 父 窗口 
制作 无 边框 的 窗口 
ADODB 介绍 及 使 用 
更 改 数据 库 时 的 注意 事项 


豆 吾 吾 吾 至 
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8.1 开发 背景 


X XX 影视 有 限 公 司 是 一 家 以 影视 为 主 的 公司 。 为 了 扩大 企业 规模 ， 增 强 企 业 的 竞争 力 ， 决 定向 
多 元 化 发 展 。 计 算 机 技术 、 网 络 通信 技术 、 多 媒体 技术 及 数据 存储 技术 的 飞速 发 展 ， 对 人 类 的 生产 和 
生活 方式 产生 了 很 大 影响 ， 网 络 传播 以 其 特有 的 快速 、 高 效 、 便 捷 的 传输 方式 直接 被 人 们 所 接受 。 随 
着 多 媒体 数据 的 存储 、 传 输 和 应 用 技术 的 不 断 成 熟 ， 以 及 宽带 网 络 的 不 断 发 展 ， 我 们 有 理由 相信 宽带 
点 播 一 定 会 成 为 网 络 内 容 创 新 的 重头 戏 。 


8.2 需求 分 析 


根据 客户 要 求 及 成 本 核算 ， 本 系统 所 要 实现 的 功能 如 下 : 

网 站 实行 会 员 等 级 管理 ， 分 “普通 会 员 ” 和 “高 级 会 员 ” 两 种 。 

网 站 提供 下 载 功能 和 在 线 视听 功能 ， 不 同 的 会 员 等 级 实现 的 功能 操作 也 不 同 。 

网 站 分 前 台 和 后 台 两 部 分 ， 没 有 权限 ， 用 户 无 法 进入 后 台 进行 管理 。 

独特 的 点 歌 模 块 ， 为 网 站 的 会 员 提 供 点 歌 平台 。 

前 期 数据 量 比较 少 ， 可 以 使 用 小 型 数据 库 以 节约 成 本 ， 如 果 后 期 数据 量 增 大 ， 则 可 以 更 换 大 
型 数据 库 。 


加 图 图 图 加 


8.3 系统 分 析 


8.3.1 系统 目标 


online 影视 365 网 主要 实现 以 下 目标 : 

自 定义 的 鼠标 样式 增强 了 网 站 界面 的 和 谐 性 。 

采用 无 边框 窗口 技术 ， 保 障 系统 的 安全 性 。 

信息 搜索 灵活 、 方 便 。 

采用 多 媒体 引用 技术 播放 影音 文件 。 

网 站 采用 文件 上 传 和 下 载 技术 实现 图 像 、 音 频 、 视 频 文 件 的 客户 端 上 传 和 下 载 。 


8.3.2 ”系统 功能 结构 


根据 online 影视 365 网 的 特点 ， 可 以 将 其 分 为 前 台 和 后 台 两 个 部 分 设计 。 前 台 主要 实现 在 线 视听 、 
影视 音乐 上 传 、 影 视 音乐 下 载 、 在 线 聊 天 和 在 线 点 歌 等 功能 ;后 台 主要 用 于 管理 员 对 影视 音乐 目录 、 
数据 信息 和 上 传 日 志 进 行 管理 。 

online 影视 365 网 的 前 台 功 能 结构 如 图 8.1 所 示 。 


| 
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online 影 视 365 网 前 台 
I 
在 线 视听 ] | 上 传 专区 | [点 歌 专区 | [详细 信息 | | 信息 展示 | [信息 查询 | | 下 载 排行 | | 会员 专区 | [ 站 长 登录 
视 | [ 音 | [点 | [点 | [ 音 || 视 视 视 | [ 音 | 会 | 会 后 
在 | | 在 | | 频 | | 频 | | 歌 | | 歌 | | 频 || 频 频 | | 一 | | 高 | | 频 | | 频 | 员 | 员 | 会 | 会 | | 台 
线 | | 线 数 | | 数 信 | | 信 信 | | 信 信 般 | | 级 下 | | 下 || 密 | 资 | 员 | 员 | | 登 
视 | | 观 | | 据 || 据 | | 息 || 息 | | 息 | | 息 息 | “| 查 || 查 | | 载 | | 载 | 码 | 料 | 注 | 登录 
听 || 看 ”| 上 || 上 | 添 || 画 | | 查 | | 查 展 | “| 询 || 询 | | 排 | | 排 | 找 | 修 | 册 | 录 | | 管 
传 | | 传 | | 加 | | 读 | | 看 | | 看 示 行 | | 行 | 回 | 改 理 


图 8.1 online 影视 365 网 前 台 功 能 结构 图 
online 影视 365 网 的 后 台 功 能 结构 如 图 8.2 所 示 。 


online 影 视 365 网 后 台 


Fr| [上 视 | 说 视 | 说 
传 | | 从 频 | | 频 | 香 | | 乱 | 管 | 闫 | | 瞻 
日 | 旧 数 | | 数 | 型 | 要 | al 旧 
志 | | 去 据 | | 握 | 吕 | | 吕 | | 品 民 | 有 
引 国 国 国 四国 四 国 攻 


图 8.2 online 影视 365 网 后 台 功 能 结构 图 
8.3.3 功能 预览 


onlie 影视 365 网 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 儿 
个 典型 功能 的 页 面 ， 其 他 页 面 参 见 光 盘 中 的 源 程序 。 

影视 专区 页 面 如 图 8.3 所 示 ， 该 页 面 展示 所 有 的 视频 信息 及 其 分 类 。 音 乐 专 区 页 面 如 图 8.4 所 示 ， 
该 页 面 展 示 所 有 的 音频 信息 及 其 分 类 。 


图 8.3 影视 专区 (光盘 \TM\08\online\list.php?action=audio) 图 8.4 ”音乐 专区 (光盘 \TM\08\online\list.php?action=video) 


@ 
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最 新 影视 页 面 如 图 8.5 所 示 , 该 页 面 用 于 显示 所 有 的 最 新 上 传 的 影视 文件 。 注 册页 面 如 图 8.6 所 示 ， 
该 页 面 显 示 了 游客 注册 时 需要 填写 的 昵称 、 密 码 和 个 人 资料 等 相关 内 容 。 


图 8.5 最 新 影视 (光盘 \TM\08\online\ list.php) 8.6 ”注册 (光盘 \TM\08\online\register.php) 


简单 查询 页 面 如 图 8.7 所 示 ， 该 页 面 显示 简单 查询 所 查 到 的 音频 、 视 频 文件 列表 。 高 级 查询 页 面 如 
图 8.8 所 示 ， 该 页 面 用 于 模糊 查询 音 、 视 频 文 件 ， 包 含 了 多 个 字段 ， 可 任意 选择 。 


图 8.7 简单 查询 (光盘 \TM\08\online\show.php) 图 8.8 ”高 级 查询 (光盘 \TM\08\online\high.php ) 


8.3.4 系统 流程 图 
online 影视 365 网 的 系统 流程 图 如 图 8.9 所 示 。 


用 户 


一 上 传记 录 查 看 
8.9 ”online 影视 365 网 的 系统 流程 图 
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8.3.5 ”开发 环境 


在 开发 online 影视 365 网 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 


1. 服务 器 端 


操作 系统 : Windows Server 2003 。 
服务 器 : Apache 2.2.8。 

PHP 软件 : PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 


开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0。 

分 辨 率 : 最 佳 效 果 为 1024x768 像素 。 
客户 端 


浏览 器 : 推荐 使 用 IE 6.0 及 以 上 版 本 。 
分 辨 率 : 最 佳 效果 为 1024x768 像素 。 
文件 夹 组织 结 构 


online 影视 365 网 包括 前 台 和 后 台 两 部 分 ， 所 
以 文件 夹 结构 也 主要 由 两 部 分 组 成 online 影视 365 
网 的 文件 夹 组 织 结构 图 如 图 8.10 所 示 。 


因 办 办 办 办 办 轨 


区 办 


8.3， 


O) 


MySQL 图 形 化 管理 软件 : phpMyAdmin-2.10.3 。 


影视 365 系 统 根 目录 
后 台 管理 系统 目录 
后 台数 据 库 链 接 文件 目录 
.后 各 ess 样式 文件 目录 
四 上 后台 背 景 及 图 标 文件 目录 
田 后 台 自 定义 函数 及 公共 文件 目录 
图 后 各 JavaScript 脚 本 文件 目录 
田 包 ADODB 库 文件 目录 
田 包 前 台数 据 库 链接 文件 目录 
=】 前 台 ess 样 式 文件 目录 
日 如 一 -一 衣 台 背景 及 图 标 目录 
四 外 in 二 前 台 自 定义 函数 及 公共 文件 目录 
田 包 js 一 前 6Jarsscript 脚 本 文件 目录 
日 wiil 上 传 文件 管理 目录 


图 8.10 ”online 影视 365 网 的 文件 夹 组 织 结构 图 


8.4 ”数据 库 设计 


8.4.1 数据 库 分 析 


本 系统 使 用 的 是 MySQL 数据 库 ， 但 使 用 的 是 ADO 连接 方式 ， 这 是 为 了 便于 以 后 当 数 据 负 担 加 重 


时 ， 只 需要 更 改 其 他 数据 库 〈 如 Oracle)， 而 无 须 重 新 编写 源 程序 。 


8.4.2 数据库 概念 设计 


通过 需求 分 析 和 功能 上 的 设计 ， 本 系统 规划 出 管理 


E 员 信息 实体 、 会 员 信 息 实 体 、 视 频 信息 实体 、 


音频 信息 实体 、 视 频 目 录 实 体 和 音频 目录 实体 。 下 面 给 出 主要 的 实体 及 E-R 图 。 
会 员 信息 实体 包括 注册 用 户 的 详细 个 人 信息 ， 如 果 想 下 载 或 在 线 视 听 ， 则 必须 注册 为 会 员 才 可 以 。 


会 员 信息 实体 E-R 图 如 图 8.11 所 示 。 


视频 信息 实体 包括 视频 名 称 、 视 频 图 片 、 视 频 文件 、 主 要 演员 、 导 演 、 发 行商 等 多 项 资料 。 视 频 


信息 实体 E-R 图 如 图 8.12 所 示 。 


图 8.11 会 员 信息 实体 E-R 图 
音频 目录 实体 包括 音频 目录 分 类 的 相关 信息 ， 如 目录 名 称 、 目 录 级 别 、 父 目录 名 称 等 。 音 频 目 录 


实体 E-R 图 如 图 8.13 所 示 。 


音频 目录 


视频 名 称 到 创建 时 间 


图 8.12 视频 信息 实体 E-R 图 图 8.13 音频 目录 实体 E-R 图 
8.4.3 数据库 物 理 结构 设计 
online 影视 365 网 的 数据 库 中 共 包 含 8 个 数据 表 ， 如 图 8.14 


创建 用 户 


所 示 图 服务 器 :localhost ， 加 数据 库 : db_online 
° 表 类 型 整理 说 明 
下 面 来 看 一 下 各 个 表 的 结构 设计 。 tb_account WYISAM。 gb2312_chinese_d 。 会员 信息 列表 


audio MyISAM 9b2312_chinese_cl ”视频 信息 表 潜 

jtb_audiolist MyISAM gb2312_chinese_ci ”视频 目录 列表 

1. tb_account (会 员 信息 列表 ) mate we ovata enna ets 
py i Re ae ee a 
会 员 信息 列表 主要 存储 用 户 的 个 人 信息 , tb_account 表 的 结构 。 sw apse mi 

如 图 8.15 所 示 。 ) Mideolist MYISAM gb2312_chinese_ci ” 圭 类 目录 列表 


2. tb_audio (视频 信 息 列表 ) 图 8.14 db_online 库 中 的 数据 库 列表 
视频 信息 列表 主要 上 传 视频 的 资料 ， 如 视频 名 称 、 视 频 图 片 等 。tb_audio 表 的 结构 如 图 8.16 所 示 。 
3. tb_audiolist (视频 目录 列表 ) 

视频 目录 列表 主要 是 上 传 视 频 所 属 的 类 型 (二 级 目录 ) 及 类 别 (一 级 目录 )。tb_audiolist 表 的 结构 
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如 图 8.17 所 示 。 


| 加 服务 器 : localhost 司 数据 库 : db_online 图 表 :tb_audio 
了 有 类型 尾 性 hul 至 这 说明 
uu in 理 自动 篇 号 
name varcher(100) go2312_chirese_ri 要 视频 名 称 
Picture varchar(200) go2312_ chinese_ri 要 封面 图 片 
加 服务 器 -1ocalhost ”加 数据 库 : db_online ， 四 表 :tp_account shes archarGs0) go2312 chinese sl 否 视频 大 小 
FR 。。 基 型 要 于 | 四 天 河 EET grace varchartD) go2312 chinese Hi 再 应 
二 一人 上 一 一 一 上 一 Publsher aehatta pal2zhmeseal 再 2 
Dassword varcharia0) 202312.chinase_cl 再 6 ba ina EN 
question 。 Yarcharl50 gu2312 chinese ci 理 冤 码 提示 问 时 ector Yarchar(100) go2312_chinese_ei 否 导演 
answer 。 Yartharl50) gb2312 chinase ci 理 客 码 棍 示 若 究 da 村 
Teatame varthar30l 。 gb2312_chinese 吕 否 二 直立 姓名 ES 加 到 种 
mambers varchar(20) gb2312 chinase_ci 理 身 做 证 号 bd Varchar(5D) 902312_chinese_ri 否 ES 
al go2312 chinese di 理 二 人 ws varehar(s0) go2312_chirese_ rl 要 关 别 
i ne) 理 全 和 新 froms varcher(100) go2312_chinese_ sl 否 视频 所 时 出 区 
和 Varehar50l ab2912_cninase cf 再 Hi 工作 PubEshTime da 否 发 有 时 则 
emal varehar00) gb2012_cninese dl 理 El remak varchar(o00) go2312_chirese_rl 百 宙 介 绍 
address varchari100) go2312_chinase_ci 理 联系 地 二 property varchar(20) 002312 chinese_ci 否 视频 层 性 
phone -varchari20) go2312.chinese_cf 理 联系 电 括 aess varchar(200) 。 gb2313_rhinese_i 百 存 能 地 直 
Vareher20l ab2312_chinose_ol 再 Fe weomame varchar(50) go2312_chirese_ i 再 发 布 人 姓名 
http vorcher(100) 952312_chinase_ci 再 个 人 主页 issueDate dateime 否 发 和 时 间 
couts mi) 再 0 Lk mrime Nk) 否 0 T 雪 站 
ade vorchari0) gb2912_chinaae_ol 再 王 进 全 只。 会员 如 lastlime dateime 否 最 后 下 载 于 间 
whether 。 varcherflol go2312_chinese_cl 雪 Et 是 本 政治 [eter varchara0) go7312ehirese_tl 否 1 是 而 新 品 


图 8.15 会 员 信 息 列 表 结 构 
4. tb_manager (管理 员 列 表 ) 


图 8.16 视频 信息 列表 结构 


管理 员 列 表 主 要 用 于 后 台 管理 员 的 资料 信息 管理 ， 包 括 管理 员 名 称 、 管 理 员 权限 等 。tb_manager 


表 的 结构 如 图 8.18 所 示 。 
中 服务 器 : localhost ， 成 数据 库 : db_online 》 贺 表 :tb_audiolist 
于 设 2 至 理 胰 性 Nul 默认 说明 
a int(4) 否 目 动 编号 
grade Varchart20) gb2312_chinese_el 理 目录 绷 别 
name Varchar(50) gb2312_chin: 香 目录 名 称 
tother varchar(s0) gb2312.chinese_dl 要 多 地 条 
userName varchar(30) gb2312_chinase_ti 理 他 娃 人 昵称 
issueDate 。 datetime 理 他 限时 间 


图 8.17 视频 目录 列表 结构 
5. tb_register (点 歌 信息 列表 ) 


阮 服务 器 : localhost 》 司 数据 库 : db_online ， 加 胡 : tb_manager 
字 有 & ES 整理 性 性 Null 默认 说 明 
地 in) 再 自动 内 时 
ame 。。 varehar(50) gb2312_ehinese_el 再 首 理 只 遇 机 
password varchar(s0) go2312_chinese_cl 殖 管理 员 更 码 
type Yarchar(50) 902312_chinese_ci 再 管理 员 类 型 
realName varchar(50) gb2312_chinese_ci 理 真实 姓名 
issueDate datetme 要 凶险 时 间 
whether 。 varchar(s0) gb2312_chinese_c 理 是 本 涡 洁 


图 8.18 管理 员 列 表 结 构 


点 歌 信 息 列 表 主要 存储 会 员 用 户 之 间 的 点 歌 祝 福 的 信息 ， 主 要 包括 发 送 人 、 接 收 人 、 歌 曲 地 址 等 。 


tb_register 表 的 结构 如 图 8.19 所 示 。 
6. tb_video (音频 信息 列表 ) 


音频 信息 列表 主要 用 于 音频 文件 的 信息 管理 ， 如 歌 
手 、 歌 词 等 。tb_video 表 的 结构 如 图 8.20 所 示 。 


7. tb_videolist (音频 目录 列表 ) 


音频 目录 列表 主要 用 于 音频 文件 的 分 类 管理 , 有 目录 
名 称 、 目 录 级 别 等 。tb_videolist 表 的 结构 如 图 8.21 所 示 。 


:localhost》 车 :db_eniine ， 加 


:由 _register 


ES 硬性 Null 实 认 说明 


intd 
intd 


varchar(50) gb2312_chineso_ci 
varchar(sD) gb2312_chineso_ei 
varchar(50) gb2312_chinese_ci 
dateime 


自动 过 号 
歌曲 电站 
并 闫 人 媳 各 
按 要 人 媳 各 
备注 

点 歌 时 间 


8.19 点 歌 信息 列表 结构 


@ 


PHP 项 目 开 发 全 程 实录 (第 3 版 ) 


:localhost 上 国 
EB 至 理 导 竹 Mul 守 久 民明 
nt) 本 Es 
Varhartl0 on2312 chnese 6 再 下 
artharat0)。 欧 2312 shinese gl 和 下 
varthart100) p2312 chinese of 要 二 喇 
varthartst) gp2312.enrese ol 至 加 作者 
varcharts0) go2312_phnese. ol 国 各 对 作者 
warthartst) 02312_Pnresa_ol 再 pF 
varehars0) go2312_chinese_ ol 再 Bg 
varchart100) 92312_chinese ci 理 故 石 疝 
varehart100) go2312 ennese ol 否 妈 
arthars0。 gb2312 chinese ol 理 Ey 
Vanarso 。 gp7317_Fnnese e 理 x 
varehar(20) go2312.phinese.e 再 Ld 
vate 再 0 
varehart1000) go2312.chinece et 再 和 
S| 下 ok 再 et 器 ER Tecahesty 局 要 锯 库 ; db_online y 司 家 ;tb_videolist 
varehar00) go2312. ennese el 否 圭 人 地 址 ee Ee 和 Er 
warehars0) go2312 chinese ol 再 省 人 几 各 pe 下 本 Pete 
aetme 百 a ame Varhag0l oh2312.einose.d 理 上 杂交 
世人 机 TE ame area ohza12.cinesed 殖 3 时 
aetme 要 最 后 下 考生 (serName varhacol ob2312_ehinesed 百 他 各 人 让 各 
Varthor(s0) 902312_ohineoo. ol 页 1 是 NW soeDate dle 百 Er 


图 8.20 音频 信息 列表 结构 图 8.21 音频 目录 列表 结构 


8.5.1 前 台 首 页 概述 


online 影视 365 网 的 前 台 首 页 的 功能 模块 主要 包括 以 下 3 部 分 。 
回 ”网 站 首部 导航 栏 : 主要 有 影视 专区 、 音 乐 专区 、 上 传 专区 和 
回 ”网 站 左 侧 导 航 栏 : 包括 会 员 登 录 模 块 、 搜索 模 决 和 影视 排行 模块。 

回 ”网 站 主 显示 区 : 包括 最 新 上 传 模块 、 视 频 模块 和 音频 模块 。 

本 案例 中 提供 的 前 台 首 页 如 图 8.22 所 示 。 该 首页 在 本 书 光盘 中 的 路 径 为 \TM\08\online\index.php。 


【 


图 8.22 online 影视 365 前 台 首 页 
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8.5.2 ”前 台 首页 技术 分 析 


在 本 章 的 数据 库 分 析 中 已 经 提 到 ， 本 系统 所 使 用 的 连接 数据 库 的 方式 为 ADODB 连接 。 这 里 ， 先 
来 初步 了 解 一 下 ADODB 如 何 连 接 MySQL。 
下 面 以 本 系统 使 用 的 数据 库 和 数据 表 为 例 ， 来 介绍 一 下 使 用 ADODB 的 基本 操作 步骤 。 
(1) 载 入 (include ) adodb.in.php 文件 。 
include "../ado/adodb.inc.php"; 


(2) 建立 数据 库 类 型 连接 。 


$conn = &ADONewConnection("mysq"); or S$conn= &NewADOConnection("mysql"); 
(3) 连接 要 使 用 的 数据 库 。 


$conn->PConnect("db_online"); or $conn->Connect("db_online"); 


(4) 执行 SQL 语句 。 
$recordset = $conn->Execute("select * from tb_audio"); 


(5) 判断 结果 ， 并 对 结果 处 理 。 


if($recordset) // 判 断 execute() 函 数 执行 中 是 否 有 错误 
while(!$recordset->EOF){ // 如 果 没 有 错误 ， 则 配合 wihle 语句 循环 输出 结果 
echo S$recordset->fields[0]."<br>"; 
S$recordset->movenext(); /! 指 针 下 移 
} 
} 
(6) 关闭 连接 。 


$recordset->close(); 
$conn->close(); 


如 果 是 对 微软 的 ADO 有 所 了 解 的 用 户 ， 会 对 上 述 步骤 有 一 种 亲切 感 。 无 论 从 字段 名 称 ， 还 是 从 功 
能 上 ， 两 者 都 非常 相似 ， 所 以 上 手 会 很 轻松 。 没 有 接触 过 ADO 的 用 户 也 不 用 心急 ， 用 ADODB 操作 数 
据 库 主要 就 是 这 6 个 步骤 ， 只 要 记 住 这 个 顺序 ， 做 几 遍 练习 后 ， 也 会 运用 自如 。 

关于 ADODB 类 库 的 详细 讲解 请 参见 8.14 节 。 


8.5.3 ”前台 首页 的 实现 过 程 


国 前台 首页 使 用 的 数据 表 : tb_account、tb_audio、tb_video 
在 首页 中 ， 使 用 到 了 3 个 include 语句 ， 将 导航 栏 、 登 录 框 、 搜 索 框 、 浏 览 区 等 主要 的 模块 加 载 进 
来 。 限 于 篇 幅 ， 这 些 页 面 的 代码 不 做 讲解 ， 请 自行 到 光盘 中 查找 。 前 台 首 页 的 实现 代码 如 下 : 
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倒 程 01 ”代码 位 置 ， 光 盘 \TM\08\online\index.php 


<?php 

session_start(); /开启 session 支持 
?> 
<script src="js/chk.js" language="javascript"></script> /引入 js 脚本 文件 
<link rel="stylesheet" href="css/style.css" /> /11 引入 css 样式 文件 
<?php 

include "top.php"; // 首 部 导航 及 LOGO 
?> 


<table border="0" cellpadding="0" cellspacing="0"> 
<tr><td width="265" align="center" valign="top"> 
<?php 
include "left.php"; /登录 、 搜 索 框 
?> 
</td><td width="605" align="center" valign="top"> 
<?php 
include "main.php"; // 主 浏览 区 
?> 
</td></tr></table> 


8.6 详细 信息 查看 模块 
8.6.1 信息 查看 模块 概述 


详细 信息 查看 模块 主要 包括 视频 详细 信息 查看 和 音频 详细 信息 查看 两 个 部 分 。 查 看 信息 功能 没有 
权限 限制 ， 无 论 是 游客 还 是 会 员 都 可 以 进行 查看 。 但 查看 页 面 中 的 下 载 和 在 线 视听 按钮 则 只 有 会 员 才 可 
以 看 到 ， 游 客 只 能 看 到 “返回 ”按钮 ， 用 来 关闭 查看 的 页 面 。 信 息 查 看 模块 的 框架 图 如 图 8.23 所 示 。 


图 8.23 ”信息 查看 框架 图 
8.6.2 ”信息 查看 模块 技术 分 析 


在 这 个 模块 中 ， 所 要 解决 的 技术 难点 就 是 : 系统 需要 根据 不 同 的 浏览 用 户 ， 显 示 不 同 的 操作 权限 。 
这 里 是 使 用 让 语句 ， 配 合 session 来 实现 其 功能 的 。 首 先 通过 检测 $_ SESSION[name] 是 否 存在 ， 来 判断 
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用 户 是 否 登 录 ， 如 果 没有 登录 ， 则 跳 过 “在 线 视听 ”和 “下 载 ” 按 钮 ， 只 显示 “返回 ”按钮 :如果 用 
户 为 登录 会 员 ， 则 显示 “在 线 播放 ”按钮 。 接 着 判断 $_SESSION[grade] 是 否 为 高 级 会 员 ， 如 果 登 录 会 
员 为 普通 会 员 ， 则 “下 载 ” 按 钮 为 灰色 失效 状态 : 如 果 登 录 会 员 为 高 级 会 员 ， 则 同时 显示 3 个 按钮 的 
全 部 功能 。 该 功能 的 关键 代码 如 下 : 
倒 程 02 ”代码 位 置 ， 光盘 \TMW8\onlinevintrophp 
<?php 
让 如 果 用 户 是 登录 会 员 */ 
if($_SESSION[Iname]<>"™" ){ 
?> 
<!-- 显 示 “ 播 放 ” 按 钮 -> 
© <input name="Submit' type="submit" value=” 播 放 " onclick="javascript:Wopen=open ('operation.php? 
action=see&id=<?php echo $rst->fields[16]; ?>",","height=700,width=665,scrollbars=no');"> 
<?php 
让 如 果 是 登录 会 员 */ 
Hf($_SESSION[name]<>"){ 
?> 
让 如 果 是 高 级 会 员 ， 则 “下 载 ” 按 钮 为 激活 状态 ， 否 则 为 灰色 */ 
@ <inputname="Submit" type="button" <?php if ($_SESSION[grade]<>" 高 级 会 员 "){ echo "disabled";}?> 
value="” 下 载 " ondlick="javascript:Wopen=location='download.php?action=audio&id=<?php echo S$rst->fields 


?> 

/显示 “返回 ”按钮 

<input name="Submit2" type="button" value=” 返 回 "class="submit" 
© onClick="javascript:top.window.close();"></td> 


< 代码 贴 二 
@ open() 方 法 可 以 打开 一 个 无 边框 的 子 窗口 。 关 于 open() 方 法 的 介绍 ， 可 以 参考 8.13 节 。 
@ disabled: 译 为 无 效 的 、 不 起 作用 的 ， 这 里 是 button 按钮 的 属性 ， 即 不 可 点 击 状态 。 
四 top.window.close0: 关闭 弹出 窗口 。 


[ TP 革 祝 365 


8.6.3 ”信息 查看 的 实现 过 程 


国 消息 管理 模块 使 用 的 数据 表 : tb audio、tb_video 

用 户 可 以 在 信息 展示 页 面 单 击 久 图 标 进入 详细 信息 
展示 页 面 。 详 细 信 息 展示 页 面 主要 是 根据 传递 的 参数 进 
行 数 据 库 检索 ， 并 将 结果 集 输出 到 浏览 器 中 。 查 看 页 面 
的 运行 结果 如 图 8.24 所 示 。 

在 详细 信息 展示 页 面 (intro.php) 中 , 通过 id 值 从 数 
据 表 中 提取 所 要 显示 的 信息 输出 到 页 面 ， 当 文件 信息 输 
出 完毕 后 , 通过 session 值 来 判断 访问 者 是 游客 还 是 会 员 ， 图 8.24 详细 信息 页 面 的 运行 结果 
如 果 是 游客 ， 则 只 显示 “返回 ”按钮 ， 如 果 是 会 员 ， 还 
要 显示 “在 线 试听 ”按钮 和 “下 载 ” 按 钮 。 程 序 的 关键 代码 如 下 : 
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倒 姓 03 ”代码 位 置 : 光盘 \TM\08\online\intro.php 


<?php 
session_start(); 
include "conn/conn.php"; 
?> 
<link rel="stylesheet" href="css/style.css"> 


< 显示 数据 详细 信息 > 


hp 
Srst = $conn->execute($sql); 

/如 果 结 果 集 不 为 空 ， 则 输出 信息 六 

if(l$rst>EOFI 

?> 


$sql="select * from tb_audio where id=".$_GETT[id]; 


/开启 session 支持 
// 载 入 数据 库 连接 文件 


/1/1 引入 css 样式 文件 


// 根 据 id 生成 查询 语句 
/| 执行 SQL 语句 ， 返 回 结果 集 


<table width="400" border="0" align="center" cellpadding="0" cellspacing="0"> 


<tr> 


<td height="15" colspan="2">&nbsp;</td> 


</tr> 
<!-- 输 出 视频 名 称 一 > 


<tr> 


<td width="131" height="20" align="right" valign="middle"> 名 称 : 


</td> 


<td width="269" height="20"><?php echo $rst->fields[1]; ?></td> 


</tr> 
<!-- 输 出 文件 大 小 一 > 


<tr> 


<td height="20" align="right" valign="middle"> 大 小 : </td> 
<td height="20"><?php echo $rst->fields[3]; ?></td> 


</tr> 


<tr> 


<td height="30" colspan="2" align="center valign="middle"> 


/根据 不 同 用 户 ， 显 示 不 同 权限 


8.7 


8.7.1 点 歌 模块 概述 

用 户 通过 主导 航 条 进入 点 歌 专区 。 在 点 
歌 专区 ， 可 以 对 专区 内 的 音乐 进行 试听 、 浏 
览 音频 信息 的 详细 内 容 和 打开 点 歌 页 面 进 
行 点 歌 , 进行 点 歌 的 前 提 条 件 是 用 户 必 须 登 
录 。 点 歌 模 块 的 用 例 图 如 图 8.25 所 示 。 


点 歌 模块 技术 分 析 
本 系统 的 在 线 点 歌 模块 实现 了 会 


8.7.2 
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点 歌 模块 设 


计 


一 一 二 


8.25 ”点 歌 模块 的 用 例 图 


视 福 语 


员 间 发 送 祝福 与 点 歌 的 功能 , 收 到 祝福 的 会 员 可 以 实现 在 线 听 歌 。 
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本 节 和 8.8 节 的 在 线 视听 模块 中 ， 将 逐一 介绍 实现 在 线 视听 的 两 种 方法 。 本 节 先 来 介绍 HTML 标签 中 
的 播放 多 媒体 标签 <embed>。 


1. 基本 语法 


<embed src=url> 


url 为 音频 或 视频 文件 ， 其 路 径 可 以 是 相对 路 径 ， 也 可 以 是 绝对 路 径 。 

<embed> 可 以 用 来 播放 各 种 多 媒体 文件 ， 格 式 可 以 是 midi、wav、mp3 等 ， 当 下 主流 的 浏览 器 都 
支持 该 标签 。 

2. 属性 设置 

<embed> 标 签 有 很 多 属性 ， 常 用 的 属性 及 说 明 如 表 8.1 所 示 。 

表 8.1 <embed> 常 用 属性 及 说 明 

属 性 名 说 了 明 举例 
tostart | 该 属性 规定 音频 或 视频 文件 是 否 在 下 载 完 之 后 就 自动 播放 。 | -bad von mp3" autostart tne 
true 表示 音乐 下 载 完成 后 自动 播放 ; false 表示 下 载 完成 后 不 播放 |“ "了 
该 属性 规定 音频 或 视频 文件 是 否 循环 及 循环 次 数 。 
loop 属性 值 为 true 时 ,音频 或 视频 文件 循环 ; 属性 值 为 false 时 ， 音 | <embed sre="1.mp3" loop=true> 
频 或 视频 文件 不 循环 。 如 果 为 正 整数 ， 则 为 循环 次 数 
i 该 属性 规定 控制 面板 是 否 显 示 ， 默 认 值 为 no。 ee 
“| true 表示 隐藏 面板 ，no 表示 显示 面板 人 
该 属性 规定 音频 或 视频 文件 开始 播放 的 时 间 。 未 定义 则 从 文件 | <embed src="lmp3" 


starttime 


开头 播放 starttime="00:10"> 
该 属性 规定 音频 或 视频 文件 的 音量 大 小 。 未 定义 则 使 用 系统 本 
volume <embed src="1.mp3" volume="10"> 
身 的 设 定 
width 该 属性 规定 了 控制 面板 的 宽度 <embed sre="1.mp3" width="100"> 
height 该 属性 规定 了 控制 面板 的 高 度 <embed sre="1.mp3" height="200"> 
title 该 属性 规定 了 音频 或 视频 文件 的 说 明文 字 <embed src="1.mp3" title="a good song"> 


除了 这 些 属性 外 ，<embed> 标 签 还 可 以 对 面板 的 外 观 及 其 背景 色 和 前 景色 进行 设置 ， 感 兴趣 的 读 
者 可 以 查找 相关 的 资料 。 


8.7.3 点 歌 模块 的 实现 过 程 


国 点 歌 模块 使 用 的 数据 表 : tb register、tb_video 

单 击 “点 歌 专区 ” 超 链接 ， 首 先进 入 到 点 歌 专区 页 面 ， 页 面 中 的 音乐 试听 和 信息 查看 在 其 他 模块 
中 都 有 介绍 ， 这 里 主要 看 一 下 点 歌 功能 。 单 击 辆 小 图 标 ， 将 进入 到 点 歌 信息 页 面 ， 点 歌 页 面 的 运行 结 
果 如 图 8.26 所 示 。 

在 点 歌 信息 页 面 ， 需 要 用 户 添加 祝福 语 和 接收 人 名 称 。 点 歌 页 面 涉 及 的 HTML 表单 元 素 如 表 8.2 
所 示 。 
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| 
点 歌 记录 详 单 后 每 记录 评 单 
接 权 人 ; [a 一 一 一 一 一 一 一 


祝 村 请， i | 
到 


[Ee 
图 8.26 点 歌 模块 的 运行 结果 
表 8.2 点 歌 页 面 涉 及 的 HTML 表单 元 素 


含义 
action="Register.asp?action=add” 表单 
| ex | ironame'sicsao | 接收 和 名称 
= ="5" id= 简要 介绍 
| Higgen | vane='<phpechos GETidlz， | 所 点 歌曲 的 地 址 
| buton | click=esisero"”， | “点 歌 ” 接 钥 


在 点 歌 页 面 ,将 前 页 传 过 来 的 歌曲 id、 接 收入 、 祝福 语 等 信息 组 成 insert 语句 存储 到 数据 库 中 ， 不 
管 保存 成 功 与 否 ， 都 将 关闭 当前 子 窗口 ， 回 到 父 窗口 中 进行 操作 。 程 序 代码 如 下 : 
倒 程 04 ”代码 位 置 ， 光盘 \TM\08\online\give.php 


<?php 
session_start(); /开启 session 支持 
if($_POST[toname] <> "“){ 1/ 如果 接 收 人 不 为 空 
$id=$_POST[d]: /歌曲 id 
$toname=$_POST[toname]; /接收 入 姓名 
$from=$_SESSION[name]; // 发 送 人 姓名 
S$remark=$_POST[remark]; /备注 信息 


<!-- 生 成 insert 语句 --> 
$sql="insert into tb_register Values(",".$id.",".$from.",".$toname.”",".$remark.",".date("Y-m-d H:i:s").")"; 
Srst = $conn->execute($sql); /| 执行 SQL 语句 

J 

<script language="javascript"> 


@ 
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<?php 

if(!($rst == false)}{ 
?> 

alert(" 点 歌 信 息 保存 成 功 "); 
<?php 

}else{ 


alert(" 点 歌 失败 "); 
<?php 
} 
?> 


yy 


top.window.close(); 
</script> 
<?php 


8.8 ”在 线 视听 模块 设计 


8.8.1 在线 视听 模块 概述 


在 线 视听 主要 包括 视频 数据 在 线 观看 和 音频 数据 在 线 收 听 两 个 部 分 。 在 不 同 的 页 面 都 可 以 直接 进 
入 视听 页 面 ， 包 括 首 页 、 搜 索 结果 页 、 内 容 介 绍 页 和 列表 页 。 但 必 
须 是 登录 会 员 才 可 以 ,游客 是 没有 权限 的 .在线 视听 流程 图 如 图 8.27 
所 示 。 


8.8.2 在线 视 昕 模块 技术 分 析 


在 8.7.2 节 中 学 习 了 如 何 使 用 <embed> 多 媒体 标签 来 播放 影音 
文件 。 本 节 来 学 习 另 一 种 方法 : <object> 标 签 。 

使 用 <objec 人 标签 来 定义 一 个 嵌入 的 对 象 。 该 对 象 向 HTML 
页 面 添 加 多 媒体 。 此 元 素 运行 插入 到 HTML 页 面 中 对 象 的 数据 和 
参数 ， 以 及 可 用 来 显示 和 操作 数据 的 代码 。<objec 人 > 标签 的 基本 图 8.27 在 线 试听 流程 图 
格式 如 下 : 


<object classid="clsid:22D6F312-BOF6-11D0-94AB-0080C74C7E95" height="68" id="MediaPlayer1" width="460"> 
<param name="ShowStatusBar" value="-1"> 
<param name="Filename" value="<?php echo $_GETI[id]?>"> 

</object> 


classid: 定义 嵌入 Windows Registry 中 或 某 个 URL 中 类 的 ID 值 , 此 属性 可 用 来 指定 浏览 器 中 
包含 的 对 象 的 位 置 。 

param: 可 定义 用 于 对 象 的 run-time 设置 。 

<objec 亿 是 微软 专门 为 IE 浏览 器 打造 的 、 可 以 扩展 外 部 应 用 程序 及 插件 的 对 象 标签 , 它 和 <embed> 


_ 国 
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的 不 同 之 处 在 于 : <object> 只 支持 以 IE 技术 为 核心 的 浏览 器 系列 ， 对 其 他 的 浏览 器 则 无 效 。 而 且 如 果 
要 使 用 <object> 来 播放 多 媒体 ， 那 么 需要 安装 相应 的 播放 插件 ， 如 RealPlayer、QuickTime 等 。 


8.8.3 ”在线 视听 模块 的 实现 过 程 


国 在 线 视听 模块 使 用 的 数据 表 : tb_audio、tb_video 
会 员 可 以 在 首页 、 信 息 展 示 页 面 或 者 详细 信息 展示 页 面 单 击 “ 观 看 ”或 “试听 ”按钮 进行 在 线 视 
听 操 作 。 页 面 运行 结果 如 图 8.28 所 示 。 


OOOO Om 


图 8.28 在线 视听 模块 的 运行 结果 
以 视频 详细 信息 展示 页 面 为 例 , 用 户 可 以 单 击 到 | 图标， 打开 视频 播放 页 面 (see.php ) 进行 在 线 观看 。 
视频 播放 页 面 主要 是 根据 传递 的 参数 进行 数据 库 检索 , 并 将 对 应 数据 的 硬盘 存储 地 址 作为 多 媒体 文件 的 引 
用 地 址 ， 当 页 面 加 载 完毕 后 ， 视 频数 据 将 自动 播放 ， 实 现在 线 观看 功能 。 在 线 播放 页 面 的 主要 代码 如 下 ; 
倒 程 05 ”代码 位 置 ， 光盘 \TM\08\online\see.php 
<?php 
session_start(); /开启 session 支持 
?> 
<body> 
<!-- 使 用 object 对 象 播放 视频 /音频 文件 --> 
<object classid="clsid:22D6F312-BOF6-11D0-94AB-0080C74C7E95" width="665" height="500" id="MediaPlayer1" > 
<!-- 设 置 自动 播放 -> 
<param name="AutoStart" value="-1"> 
<!-- 设 置 状态 栏 -> 
<param name="ShowStatusBar" value="-1"> 
< 上 -播放 文件 路 径 -> 
<param name="Filename" value="upfiles/audio/<?php echo $_GET[id] ?>"> 
</object> 


</body> 
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8.9 ”后台 首页 设计 


8.9.1 后 台 首 页 概述 


根据 用 户 对 各 个 功能 模块 的 使 用 频率 和 重要 程度 , 本 系统 的 首页 面 中 要 显示 的 模块 主要 有 两 部 分 ， 
分 别 介绍 如 下 。 
(1) 网 站 左 侧 导 航 栏 : sesh rte 
目录 管理 模块 ， 主 要 包括 视频 目录 管理 和 音频 目录 管理 两 个 部 分 。 
数据 管理 模块 : 主要 包括 视频 数 WaS 
会 员 管理 模块 : 包括 会 员 等 级 管理 、 会 员 信息 冻结 、 会 员 信息 删除 3 个 部 分 。 
i 主要 用 于 浏览 和 查询 用 户 上 传 操作 的 详细 内 容 。 
员 设 置 模块 ， 主 要 包括 管理 员 信 息 添加 、 删 除 、 冻 结 3 个 部 分 。 
(2) Fe 显示 各 个 模块 的 操作 及 结果 。 
本 案例 中 提供 的 后 台 首 页 如 图 8. 9 玄 首 Ea 六 污 级 中 的 路 径 为 IMD8onlinewadmnin index.php。 


办 办 办 轨 


着 的 阅 有 学 和 
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EE 


1 看 于 全 列 。。320 


图 8.29 online 影视 365 网 后 台 首页 


8.9.2 ”后台 首页 技术 分 析 


本 系统 的 后 台 首 页 采用 的 布局 结构 为 二 分 栏 布局 。 左 侧 导航 栏 清楚 、 明 白地 显示 了 后 台 管 理 员 所 
能 使 用 到 的 功能 。 当 单 击 任意 功能 按钮 时 ， 在 主 显示 区 显示 对 应 的 操作 界面 和 该 功能 模块 下 的 子 功能 。 
页 面 简练 、 结 构 清 晰 、 浏 览 方便 ， 二 分 栏 的 主要 特点 被 表现 得 淋漓 尽 致 。 关 于 二 分 栏 的 详细 介绍 ， 请 
参见 3.6.3 节 


8.9.3 后台 首 页 的 实现 过 程 


国 后台 首页 使 用 的 数据 表 : tb_ manager 
下 面 来 看 一 下 后 台 首 页 的 实现 过 程 。 在 首页 代码 中 ,对 每 项 功能 权限 的 判断 都 是 在 left.php 中 完成 
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的 ， 只 要 在 左 侧 导航 栏 的 位 置 载 入 left.php 即 可 。 在 主 显示 区 ,根据 参数 返回 的 不 同 值 ， 显 示 不 同 的 功 
能 界面 。 程 序 的 关键 代码 如 下 : 
倒 答 06 ”代码 位 置 ， 光盘 \TM\08\online\admin\main.php 


<?php 
session_start(); /开启 session 支持 
include "inc/chec.php"; // 载 入 权限 检查 文件 
include "conn/conn.php"; // 载 入 数据 库 链 接 文件 
?> 
<link href="css/style.css" rel="stylesheet"/> /引入 css 样式 文件 
<script src="js/admin_js.js" language="javascript"></script> /引入 js 脚本 文件 
<center> 
<?php 
/* 载 入 网 站 左 侧 导 航 栏 */ 
include "left.php"; 
?> 


<div style="height:35px;">&nbsp;</div> 
<?php 
/* 根 据 不 同 的 参数 ， 显 示 不 同 的 功能 界面 */ 
if(isset($_GET[action])X 
switch ($_GET[action]){ 
case "audioList": // 视 频 目 录 添 加 
include "a_list.php"; 
break; 
case "videoList": /音频 目录 添加 
include "v_list.php"; 
break; 
case "audio": // 视 频数 据 添加 
include "audio.php"; 
break:; 
case "video": // 音 频数 据 添加 
include "video.php"; 
break; 
case "grade": // 会 员 等 级 设置 
include "grade.php"; 
break; 
case "member"; /人 会员 数据 设置 
include "member.php"; 
break; 
case "log"; /上 传 日 志 管理 
include "log.php"; 
break; 
case "manager"; // 管 理 员 设置 
include "manager.php"; 
break; 
从 
> </td></tr> 
<tr><td height="15" bgcolor="#fOf0f0">&nbsp;</td></tr> 
</table> 
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</td></tr> 
</table> 
</center> 


8.10 目录 管理 模块 设计 


8.10.1 目录 管理 模块 概述 


目录 管理 模块 主要 包括 视频 目录 管理 和 音频 目录 管理 两 个 部 分 。 管 理 员 可 以 通过 后 台 管 理 导航 进 
入 对 应 的 目录 管理 页 面 。 

如 果 管 理 员 进入 了 视频 目录 管理 页 面 (audiolist.php), 那么 在 该 页 面 内 可 以 打开 目录 添加 页 面 进行 
目录 添加 操作 ， 也 可 以 删除 相应 的 目录 。 鉴 于 目录 信息 比较 简短 ， 本 例 中 没有 提供 目录 修改 的 功能 。 
目录 管理 模块 的 用 例 图 如 图 8.30 所 示 。 


一 一 p>、 删除 音频 目录 ) < 一 一 一 一 ( 添加 视频 目录 ) 4 一 一 
一 一 六 (添加 音频 目录 ) 一 一 一 一 > (删除 视频 目录 ) 所 一 一 
音频 目录 管理 员 超级 用 户 视频 目录 管理 员 


8.30 ”视频 目录 管理 用 例 图 


4 
SE 税 明 由 于 视频 目录 管理 和 音频 目录 管理 实现 的 方法 基本 类 似 ， 因 此 ， 林 齐 重 点 讲解 视频 目录 
管理 模块 实现 的 方法 ， 关 于 音频 目录 管理 模块 的 实现 方法 请 参见 本 书 附 赠 源码 光盘 。 


8.10.2 ”目录 管理 模块 技术 分 析 


在 本 系统 中 ， 有 个 功能 经 常 要 被 用 到 : 就 是 在 添加 目录 、 歌 曲 时 ， 经 常会 碰 到 重 名 的 问题 。 对 于 
这 个 问题 ， 可 以 写 一 个 自 定义 函数 ， 并 将 它 放 到 单独 的 一 个 文件 中 ， 方 便 其 他 页 面 调 用 ， 如 本 系统 中 
的 /inc/func.php， 就 是 专门 用 来 存储 自 定义 函数 的 文件 。 

下 面 来 看 一 下 自 定义 函数 is_chk()。 

倒 程 07 ”代码 位 置 ， 光盘 \TM\08\online\admin\inc\func.php 


1/ 判断 目录 名 是 否 重复 

1/$f_fields: 字段 名 

//$tablename: 数据 表 名 

//$f_str: 要 查找 的 字段 

function is_chk($f_fields,$tablename, $f_strX{ 


$conn = &ADONewConnection(mysql); // 建 立 MySQL 连接 
$conn->PConnect("localhost","root","root","db_online"); /连接 db_online 数据 库 
S$is_chk = true; /声明 一 个 boolean 型 变量 
Sis_sqlstr = "select $f_fields from $tablename"; /查找 字段 
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Sis_rst = $conn->execute(S$is_sqlstr); 
while(!$is_rst->EOF){ /| 循环 输出 记录 集 
ereeeeseses 一 一 对 比 字段 中 的 值 sesesesssssal 
if($f_str == $is_rst->fields[O]X 
/如 果 字 段 在 数据 库 中 找到 ， 说 明 重 名 了 ，is_chk 设置 成 false*/ 


Sis_chk = false; 
break; 
} 
S$is_rst->MoveNext(); /将 指针 指向 下 一 条 记录 
return $is_chk; /| 返回 变量 $is_chk 


8.10.3 ”目录 管理 的 实现 过 程 


国 目录 管理 使 用 的 数据 表 : tb_audiolist、tb_videolist 

单 击 左 侧 导航 栏 中 的 “视频 目录 管理 ” 超 链接 ， 打 开 视 频 目录 管理 页 面 ， 在 该 页 和 
录 添 加 ” 超 链 接 、 所 有 的 一 、 二 级 目录 信息 和 对 应 的 “删除 ”按钮 。 视 频 目 录 管 理 页 面 的 运行 结果 如 
图 8.31 所 示 。 
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图 8.31 视频 目录 管理 的 运行 结果 
视频 目录 管理 页 面 的 代码 很 简单 : 使 用 while 循环 将 数据 库 中 的 记录 全 部 显示 出 来 , 并 在 每 输出 一 
条 记录 后 ， 添 加 一 个 “删除 ”操作 。 目 录 管 理 页 面 的 实现 代码 如 下 : 
倒 程 08 ”代码 位 置 ， 光盘 \TM\08\online\admin\a_list.php 


<?php 
session_start(); /开启 session 支持 
include "conn/conn.php"; /| 载 入 数据 库 链 接 文件 
include "inc/chec.php"; // 载 入 权限 检查 文件 


/* 输 出 视频 目录 数据 表 中 的 全 部 数据 */ 
$l_sqlstr = "select * from tb_audiolist"; 
$l_rst = $conn->execute($|_sqlstr); 


@ 
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<tr> 
让 目录 添加 */ 
<td height="10" colspan="5" align="right" valign="middle"><a href="#" onclick="javacript: 


Wopen= open ('operation.php? action= audiolist",' 添 加 目录 ','height=500,width=665,scrollbars=no');"> 目 
录 添 加 </a></td> 


</tr> 
<!-- 显 示 目录 相关 信息 的 字段 名 --> 
<tr> 
<td height="30" align="center" valign="middle">ID</td> 
<td height="30" align="center" valign="middle"> 等 级 </td> 
<td height="30" align="center" valign="middle"> 名 称 </td> 
<td height="30" align="center" valign="middle"> 父 级 名 称 </td> 
<td height="30" align="center" valign="middle"> 操 作 </td> 
</tr> 
<?php 
while(l$l_rst->EOF){ 
和 
<tr> 
<!- 输 出 目录 id--> 
<td height="18" align="center valign="middle"><?php echo $I_rst->fields[0]; ?></td> 
<!-- 输 出 目录 等 级 --> 
<td height="18" align="center" valign="middle"><?php echo $I_rst->fields[1]; ?></td> 
<!-- 输 出 目录 名 称 --> 
<td height="18" align="center" valign="middle"><?php echo $I_rst->fields[2]; ?></td> 
<!-- 输 出 目录 父 级 名 称 --> 
<td height="18" align="center valign="middle"><?php echo $I|_rst->fields[3] ?></td> 
<!-- 输 出 “删除 ” 超 链接 --> 
<td height="18" align="center" valign="middle"><a href="del_list_chk.php?action= audiolist&id= 


<?php echo $l_rst->fields[0]; ?>" onclick="return del_chk();"> 删 除 </a> 


</td> 
</tr> 
<?php 
/指针 下 移 一 条 记录 
$l_rst->MoveNext(); 
和 


2 


</table></td> 


</tr> 


</table></td> 


</tr> 


</table> 
8.10.4 ”目录 添加 的 实现 过 程 


国 ” 目录 添 加 使 用 的 数据 表 : tb_audiolist、tb_videolist 
单 击 视频 目录 管理 页 面 中 的 “目录 添加 ”按钮 ， 将 打开 目录 添加 页 面 。 目 录 添 加 页 面 的 运行 结果 


如 图 8.32 所 示 。 
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图 832 视频 目录 添加 的 运行 结果 


视频 目录 添加 页 面 (audiolistphp) 中 主要 包含 一 个 添加 目录 表单 ， 表 单 的 主要 元 素 如 表 8.3 所 示 。 
表 8.3 ”视频 目录 添加 页 面 涉及 的 HTML 表单 元 素 
名 称 重要 属性 含 义 
list method="post" action="audiolist_ chk.php" 视频 目录 添加 表单 
name="grade" OnChange="check()" 
grade | <option value="1" selected> 一 级 目录 </option> 目录 等 级 
<option value="2"> 二 级 目录 </option> 
<2php 
$L sqlstr = "select * from tb_audiolist where grade = "1"™"; 
$l_rst = $conn->execute($]_sqlstr); 
while(!$l_rst->EOF){ ?> 
it <option Vane Stphy echo $1_rst->fields[2]; ?>"><?php echo $l_rst-> 父 级 目录 名 称 
fields[2]; ?></option> 
<2php 
$l_rst->MoveNext(); 
} 
7> 
names, [text | ype="text" id="names" 目录 名 称 
Submit |isnbmint | class="submit" value=" 添 ”加 " onclick="return n_chkO:" “添加 ”按钮 
ee | class="submit" value=" 返 回 “返回 ”按钮 


onClick="javascript:top.window.close! 


当 管理 员 添 加 信息 提交 后 ， 系 统 转 到 处 理 页 中 (audiolist_chk.php) 进行 添加 操作 。 首 先 判断 输入 
目录 名 称 是 否 与 已 存在 的 名 称 重复 ， 如 果 重 复 ， 回 到 上 一 步 ， 如果 不 重复 ， 则 向 数据 表 中 添加 新 记录 。 
视频 目录 添加 的 程序 代码 如 下 : 

倒 程 09 ”代码 位 置 ， 光盘 \TM\08\online\admin\audiolist_chk.php 


<?php 


@ 


session_start(); // 开 启 session 支持 
include "inc/chec.php"; // 载 入 权限 检查 文件 
include "conn/conn.php"; // 载 入 数据 库 连接 文件 
include "inc/func.php"; // 载 入 自 定义 函数 文件 


/使 用 自 定 义 函 数 is_chk()， 判 断 用 户 名 是 否 重复 */ 
if(is_chk("name","tb_audiolist",$_POSTInames]) == falseX{ 


第 8 章 online 影视 365 网 (Apache+PHP+ADODB+phpMyAdmin+MySQL 5.0 实现 ) 


echo "<script>alert(' 名 称 重 复 ');history.go(-1);</script>"; // 如 果 重 复 ， 退 到 上 一 步 操作 
exit(); 


} 
/生成 insert 插入 语句 */ 
$a_sqlstr = "insert into tb_audiolist (grade,name,fatheruserName,issueDate) 
values('$_ POST[grade]','$_POST[names]','$_POSTIfather]','$_SESSION[admin]',".date("Y-m-d H:i:s").")"; 
/判断 返回 结果 ”/ 
if($conn->execute($a_sqlstr) == false) 
人 * 如 果 添加 失败 ， 返 回 上 一 步 */ 
echo "<script>alert(' 添 加 失败 ');history.go(-1);</script>"; 
else 
让 如 果 添 加 成 功 ， 则 刷新 父 窗口 ， 关 闭 子 窗口 */ 
echo "<script>top.opener.location.reload();alert(' 添 加 成 功 ');window.close();</script>"; 
?> 


8.10.5 ”目录 删除 的 实现 过 程 


国 ”目录 删除 使 用 的 数据 表 : tb_audiolist、tb_videolist 
视频 目录 的 删除 功能 是 根据 传递 的 目录 id 号 将 数据 库 中 对 应 的 记录 找到 , 并 删除 。 程序 代码 如 下 : 
倒 程 10 ”代码 位 置 ; 光盘 \TM\W8\onlineadminvdel_list_chk.php 


<?php 

Session_start(); // 开 启 session 支持 

include "inc/chec.php"; // 载 入 权限 检查 文件 

include "conn/conn.php"; /| 载 入 数据 库 链接 文件 

if($_GET[action] == "audiolist") // 如 果 action 等 于 audiolist 
$t_name = "tb_audiolist"; // 要 访问 的 数据 表 就 为 tb_audiolist 

else if($_GET[action] == "videolist") // 如 果 action 等 于 videolist 
$t_name = "tb_videolist"; /那么 数据 库 表 就 为 tb_videolist 

$sqlstr = "delete from ".$t_name." where id =".$_GET[id]; /根据 id 号 ， 删 除 对 应 记录 


* 如 果 删 除 成 功 ， 则 返回 到 当前 页 ， 否 则 ， 退 到 上 一 步 ”/ 
if($rst = $conn->execute($sqlstr) == false){ 
echo "<script>alert(' 删 除 错误 !".$sqlstr.");history.go(-1);</script>"; 


jelse 
echo "<script>alert(' 删 除 成 功 );location=".$_SERVER[HTTP_REFERER].";</script>"; 
Pn 


8.11 数据 管理 模块 设计 


8.11.1 数据 管理 模块 概述 


入 对 应 的 数据 管理 页 面 ， 如 视频 数据 管理 页 面 (audiophp)。 在 数据 管理 页 面 ， 用 户 可 以 打开 数据 添加 
页 面 或 删除 对 应 的 数据 。 数 据 管理 模块 用 例 图 如 图 8.33 所 示 。 
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音频 目录 管理 员 超级 用 户 视频 目录 管理 员 


CD 


图 8.33 ”数据 管理 模块 用 例 图 


/. 
说 明 由 于 视频 数据 管理 和 音频 数据 管理 实现 的 方法 基本 类 似 ， 因 此 ， 本 章 重 点 讲解 视频 数据 
管理 模块 实现 的 方法 ， 关 于 音频 数据 管理 模块 的 实现 方法 请 参见 本 书 附 赠 源码 光盘 。 


8.11.2 ”数据 管理 模块 技术 分 析 


在 本 模块 中 ， 主 要 运用 的 就 是 文件 上 传 技术 。 关 于 文件 上 传 方面 的 知识 ， 请 参见 本 书 8.7 节 ， 里面 
有 对 预定 义 变量 $_FILES 的 详尽 介绍 ， 和 如 何 判断 图 片 类 型 和 大 小 的 相关 方法 。 

在 这 个 模块 中 ， 略 有 不 同 的 是 使 用 了 一 个 自 定义 函数 来 判断 文件 名 后 缀 。 下 面 来 看 看 这 个 函数 的 
实现 过 程 ， 代 码 如 下 : 

倒 程 11 代码 位 置 ; 光盘 \TM\08\online\admin\inc\func.php 


// 判 断 文 件 后 缀 

//$f_type: 人 允许 文件 的 后 缀 类 型 
1/$f_upfiles: 上 传 文件 名 

function f_postfix($f_type, $f_upfiles){ 


Sis_pass = false; /设置 一 个 boolean 型 变量 $is_pass 

oo S$tmp_upfiles = split("\.",$f_upfiles); // 拆 分 上 传 文件 名 ， 存 储 到 变量 $tmp_upfiles 中 

© S$tmp_fix = $tmp_upfiles[count($tmp_upfiles) - 1]; // 取 得 上 传 文件 名 后 缀 
for($num = 0; $num < count($f_type);$num++}{ /使 用 for 循环 ， 输 出 允许 使 用 的 后 缀 类 型 

© if(strtolower($tmp_tix) == $f_type[$num])  ”// 如 果 上 传 文件 后 缀 包括 在 其 中 ， 说 阴 类 型 相符 

Sis_pass =true; //$is_pass 变量 赋值 为 true 
return $is_pass; // 返 回 变量 $is_pass 
} 
4 人) 代码 贴 二 


@ split("\.",$f_upfiles): splitO 函 数 为 字符 分 割 函数 ， 它 有 两 个 参数 ， 分 别 为 分 隔 符 和 要 分 割 字符 串 。 返 回 结果 为 分 
害 后 的 字符 事 数 组 。 例 如 ，S$tmp=split("c","placepublicslice")， 那 么 Stmp 值 就 为 Stmp[0]="plac"、S$tmp[1]="epubli"、 
S$tmp[3]="sli"、S$tmp[4]="e"。 

四 count($tmp_upfiles): countO 函 数 的 作用 是 取得 数组 个 数 。 


@ 
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目 strtolower($tmp_tix): strtolower0 函 数 返 回 参 数字 符 囊 的 小 写 格式 ， 如 strtolower("AbC")， 输 出 等 于 abc。 


eh 
说 明 尽 可 能 地 使 用 PHP 函数 库 的 函数 。 因 为 自己 写 的 函数 ， 无 论 是 从 算法 设计 ， 还 是 从 功能 
实现 上 ， 都 无 法 和 PHP 自 带 的 函数 相 比较 。 不 要 试 着 去 找寻 它们 的 错误 ， 这 是 非常 不 明智 的 做 法 。 


8.11.3 ”数据 添加 的 实现 过 程 


国 ” ”数据 添 加 使 用 的 数据 表 : tb audio、tb_video 

管理 员 可 以 通过 单 击 视频 数据 管理 页 面 的 “数据 添加 ”按钮 ， 打 开 数 据 添加 页 面 (audio.php), 在 
数据 添加 页 面 填写 了 表单 之 后 ， 将 表单 提交 给 本 页 ， 本 页 会 根据 提交 的 表单 数据 在 数据 库 中 添加 对 应 
的 记录 。 数 据 添加 页 面 的 运行 结果 如 图 8.34 所 示 。 


医 到 >esz, [本 
图 8.34 ”视频 数据 添加 页 面 的 运行 结果 
在 视频 数据 添加 页 面 中 涉及 的 HTML 表单 元 素 如 表 8.4 所 示 。 
表 8.4 视频 数据 添加 页 面 涉 及 的 HTML 表单 元 素 


名 称 重要 属性 
name="list" method="post" action="dataadd_chk.php" 


List 
enctype="multipart/form-data” 

Names, id="names" size="30" 

Picture id="picture" size="15" 

Address id="address" size="15" 


<option value=" 一 级 "> 一 级 </option> 

<option value=" 二 级 "> 二 级 </option> 

Grade <option value=" 三 级 "> 三 级 </option> 

<option value=" 无 限制 级 " selected="selected"> 无 限制 级 </option> 
<option value=" 禁 片 "> 禁片 </option> 

Publisher id="publisher" size="30" 
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名 称 | 类 型 


Actor 


id="actor" size="30" 


Director 


id="director" size="30" 


Maker 


id="marker" sizt 


Language 


<input ="radio" name="language" value=" 中 文 " checked> 中 文 
<input type="radio" name="language" value=" 英 文 "> 英文 

<input type="radio" nam 

<input type="radio" name="language" value=" 日 语 "> 日 语 

<input type="radio" name="language" value=" 德 语 "> 德语 

<input type="radio" name="language" value= "法语 "> 法 语 * 
</td> 


语种 


Style 


Types 


From text 


<?php 
S$a_sqlstr="select * from tb_videoList where grade= 27"; 
S$a_rst = $conn->execute($a_sqlstr); 
while(!$a_rst->EOF)!{ 
Js 
<option value="<?php echo $a_rst->fields[2]; ?> "><?php echo $a_rst-> 
fields[2]; ?></option> 
<?php 
S$a_rst->movenext(); 


: 


<?php 

S$t_sql = "select * from tb_audiolist where grade = “1°"; 
$t_rst = $conn->execute($t_sql); 

while(!S$t_rst->EOF){ 

人 > 

<option value="<?php echo $t_rst->fields[2]; ?> "><?php echo $t_rst-> 
fields[2]; ?></option> 

<2php 

S$t_rst->movenext(); 

} 

7> 


id="from" size="30" 


类 别 


类 型 


发 行 国家 


_publishtime text 


loadCalendar(list.publishtime); 


发 行 时 间 


news radio 


<input name="news" type="radio" value="1" checked> 是 
<input name="news" type="radio" value="0"> 否 


新 品 参数 


remark textarea 


class="submit" onclick="retum a_chkO:” 


简要 介绍 


Submit button 


class="submit" onclick="retum a_chkO:” 


“添加 ”按钮 


Submit2 button 


OnClick="j 


“返回 ” 按 角 


视频 管理 处 理 页 (dataadd_chk.php) 主要 分 3 部 分 : 图 片 文件 格式 及 图 片 大 小 判断 、 视 /音频 格式 
及 文件 大 小 判断 ， 信 息 都 准确 无 误 后 ， 则 向 数据 库 中 添加 新 记录 。 处 理 页 处 理 的 程序 代码 如 下 
倒 程 12 代码 位 置 ， 光盘 \TM\08\online\admin\dataadd_chk.php 


他 
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<?php 
session_start(); // 开 启 session 支持 
include "inc/chec.php"; // 载 入 权限 检查 文件 
include "conn/conn.php"; // 载 入 数据 库 链接 文件 
include "inc/func.php"; // 载 入 自 定义 函数 文件 
$p_type = array("jpg","jpeg","bmp","gif"); /允许 上 传 的 图 片 文件 格式 
$f_type = array("avi","rm","rmvb","wav","mp3","mpg"); // 允 许 上 传 的 视频 、 音 频 格式 
$audio_path = "../upfiles\audio"; // 视 频 上 传 位 置 
$video_path = "../upfiles\\video"; /音频 上 传 位 置 
$picture_path =""; /| 图片 上传 路 径 
S$file_path = ""; /文件 上 传 路 径 


在 图 片 类 型 判断 中 ,首先 判断 图 片 大 小 是 否 在 允许 范围 之 内 , 接着 使 用 自 定义 函数 f_postfix() 来 判断 
图 片 后 级 是 否 符合 要 求 ， 如 果 没 有 问题 ， 则 根据 timeO 函 数 生成 图 片 名 ， 并 保存 到 变量 $picture_path 中 。 
倒 竹 13 ”代码 位 置 ， 光 盘 \TM\08\online\admin\dataadd_chk.php 


/判断 图 片 大 小 ， 是 否 大 于 0KB， 小 于 700KB*/ 
© if$_FILES[picture][size] > 0 and $_FILES[picturel][size] < 700000){ 


/* 如 果 图 片 格式 正确 */ 

@ if(($postf = f_postfix($p_type,$_FILES[picture][name])) != false){ 
/使 用 time() 函 数 生成 文件 名 % 

目 S$picture_path = time().".".$postf; 


/* 如 果 action 值 为 a 时 ， 说 明 上 传 为 视频 图 片 ”/ 
if($_POST[action] == "a" 
/* 如 果 生 成 的 临时 文件 正确 ， 则 使 用 move_uploaded_file() 函数 上 传 %/ 
if($_FILES[picture][tmp_name]) 
9 move_uploaded_file($_FILES[picture][tmp_name],$audio_path."\".$picture_path); 

else{ 
echo "<script>alert( 上 传 图 片 失败 ! ');history.go(-1);</script>"; 
exit(); 


} 
/* 如 果 上 传 图 片 为 音频 图 片 */ 
}Jelse if($_POST[action] == "v"){ 
if($_FILES[picture][tmp_name]) 
move_uploaded _file($_FILES[picture][tmp_name],$video_path."\".$picture_path); 
else{ 
echo "<script>alert( 上 传 图 片 失败 ! ');history.go(-1);</script>"; 
exit(); 


} 
} 
}else{ 
echo "<script>alert( 上 传 图 片 格式 错误 ! ');history.go(-1);</script>"; 
exit(); 


”如 果 图 片 大 于 700KB， 说 明 图 片 太 大 ， 超 出 要 求 ， 返 回 上 一 步 */ 
}else if($_FILES[picturej[size] > 700000X{ 
echo "<script>alert( 上 传 图 片 大 小 超出 范 
exit(); 


");history.go(-1);</script>"; 


} 
else{ 
S$picture = "™"; 
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} 
frr en 


< 代 码 贴 十 

@ 使 用 预定 义 变量 的 $_FILES[namej[size] 属 性 判断 上 传 文件 的 大 小 。 如 果 文 件 大 于 pho.ini 中 设置 的 上 传 文件 的 最 
大 值 ， 那 么 size 将 永远 返回 0， 而 不 是 文件 的 实际 大 小 。 

@ 使 用 自 定义 函 数 f_postfix() 判 断 上 传 文件 的 后 缓 。 

目 返回 当前 UNIX 的 时 间 蕉 。 

@ 使 用 move_uploaded file0 上 传 函 数 要 注意 : 它 的 第 一 个 参数 是 上 传 到 服务 器 中 的 临时 文件 名 ， 而 不 是 文件 原始 
名 称 ， 是 用 户 自 定义 的 名 称 。 要 取得 临时 文件 名 使 用 $_FILES[name][tmp_name]. 


对 上 传 文件 的 判断 和 上 传 图 片 的 流程 基本 相同 ， 也 是 先 判断 文件 大 小 是 否 符合 上 传 文件 的 范围 要 
求 ， 接 着 判断 上 传 文件 后 级 ， 最 后 使 用 time() 函 数 生成 文件 名 ,保存 到 变量 $address_path 中 。 下 面 是 判 
断 视频 文件 的 流程 ， 代 码 如 下 : 

倒 程 14 ”代码 位 置 ， 光盘 \TM\08\online\admin\dataadd_chk.php 


让 判断 上 传 文件 类 型 与 大 小 */ 
if($_FILES[address][size] > 0){ 1// 判 断 上 传 文件 大 小 
if($_POST[action] == "a"){ // 判 断 是 视频 文件 还 是 音频 文件 
if($_FILES[address][size] < 300000000X( /| 判断 文件 是 否 超出 限定 大 小 
if(($postf = f_postfix($f_type,$_FILES[address][name])) != falseX{ 
$file_path = time().".".$postf; /| 生成 新 的 文件 名 
if($_FILES[address][tmp_name]) /| 如果 临时 文件 正确 


/使 用 move_uploaded_file() 函数 上 传 文件 */ 
move_uploaded file($_FILES[address][tmp_name],$audio_path."\".$file_path); 
else{ 
echo "<script>alert(' 上 传 文件 错误 ! ');history.go(-1);</script>"; 
exit(); 
} 
Jelse{ 
echo "<script>alert( 上 传 文件 格式 错误 ! ');history.back(-1);</script>"; 
exit(); 
Delse{ 
echo "<script>alert(' 上 传 文件 大 小 错误 ! ');history.go(-1);</script>"; 
exit(); 


人 
/* 如 果 是 音频 文件 */ 
else if($_POST[action] == "v"){ 
/判断 文件 大 小 */ 
if($_FILES[addressJ[size] < 10000000){ 
/判断 文件 后 缀 */ 
if(($postf =f_postfix($f_type,$_FILES[address][name])) != falseX{ 
S$file_path = time().".".$postf; 
/判断 临时 文件 是 否 生成 */ 
if($_FILES[address][tmp_name]) 
/* 使 用 move_uploaded_file() 函数 上 传 文件 :/ 
move_uploaded file($_FILES[address][tmp_name],$video_path."\".$file_path); 
else{ 
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echo "<script>alert(' 上 传 文件 错误 ! ');history.go(-1);</script>"; 


echo "<script>alert( 上 传 文件 格式 错误 ! ');history.back(-1);</script>"; 


echo "<script>alert(' 上 传 文件 大 小 错误 ! ');history.go(-1);</script>"; 
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exit();}else{ 
exit();}}else{ 
exit();} 
}else{ 


echo "<script>alert(' 没 有 上 传 文件 或 文件 大 于 300MB');history.go(-1);</script>"; 


exit(); 
} 


如 果 上 传 图 片 和 上 传 的 文件 都 没有 问题 ， 则 对 表单 的 其 他 信息 进行 处 理 ， 即 生成 insert 插入 语句 ， 


添加 新 记录 。 程 序 代 码 如 下 : 


倒 程 15 ”代码 位 置 ， 光盘 \TM\08\online\admin\dataadd_chk.php 


* 视 频 文件 和 音频 文件 相同 的 信息 */ 
$names = $_POST[names]; 
4$grade = $_POST[grade]; 

$sizes = $_FILES[address][size]; 
$publisher = $_POST[publisher]; 
$actor = $_POST[actor]; 
Slanguage = $_POSTI[language]; 
S$style = $_POST[style]; 

$types = $_POST[types]; 

$froms = $_POST[from]; 
$publishtime = $_POST[publishtime]; 
Snews = $_POST[Inews]; 

S$remark = $_POST[remark]; 


Pr 
if($_POST[action] == "a")( 
/* 确 认 父 级 目录 */ 
/* 表 单 值 ， 视 频 文件 特有 的 属性 */ 
$director = $_POSTI[director]; 
$marker = $_POST[marker]; 
/* 生 成 insert 语句 */ 


// 视 频 名 称 
// 级 别 
/大 小 

/发 行商 

// 演 员 

// 语 言 

1/ 类别 

// 类 型 

// 发 行 国家 
// 首 映 时 间 
// 是 否 为 新 片 
// 备 注 


// 提 取 导 演 
/| 制 片 人 


$a_sqlstr = "insert into tb_audio (name,picture,sizes,grade,publisher,actor,director,marker,languages,type, 
style,froms,publishtime,bool,remark, property,address,username,issueDate) values('‘$names','$picture_path','$sizes', 
'$grade','$publisher','$actor, '$director', '$marker','$language','$types', '$style','$from', '$publishtime','$news','remark’, ' 
管理 员 ','$file_path','$_SESSION[admin]',".date("Y-m-d H:i:s").")"; 


} 

else if($_POST[action] == "v"X 
$actortype = $_POST[actortype]; 
$ci= $_POSTIcil; 
$qu = $_POST[qu]; 
/* 生 成 insert 语句 */ 


// 如 果 上 传 文件 为 音频 文件 
/歌曲 类 型 

/作词 

/作曲 


$a_sqlstr = "insert into tb_video (name,picture,actor,ci,qu,actortype,type,style,publisher,froms,sizes, 
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languages, publishTime,remark, property,address,userName,issueDate) values('$names,','$picture_path', '$actor, 
'$ci,'$qu','$actortype','$types','$style','$publisher','$froms','$sizes','$language','$publishtime','$remark',' 管理 员 ',$file_ 
path','$_SESSION[admin]',".date("Y-m-d H:i:s").")"; 
jelse{ 
echo "<script>alert(' 错 误 ');window.close();</script>"; 
exit(); 


|; 
/上 传 图 片 和 文件 */ 
PE 


$a_rst = $conn->execute($a_sqlstr); 
if(!($a_rst == false)) 

echo "<script>top.opener.location.reload();alert(' 添 加 成 功 ');window.close();</script>"; 
else 

echo "<script>alert(' 添 加 失败 ');history.go(-1);</script>"; 


8.11.4 ”数据 删除 的 实现 过 程 


国 ”数据 删除 使 用 的 数据 表 : tb_audio、tb_video 

视频 数据 删除 主要 是 根据 传递 的 文件 id 值 , 将 数据 库 中 的 记录 删除 , 同时 使 用 unlink() 函 数 删 除 服 
务 器 端的 真实 数据 。 程 序 代码 如 下 : 

倒 程 16 ”代码 位 置 ， 光盘 \TM\08\online\admin\del_video_chk.php 


<?php 
session_start(); // 开 启 session 支持 
include "conn/conn.php"; // 载 入 数据 库 链接 文件 
include "inc/chec.php"; // 载 入 权限 判断 文件 
S$file_path = "../upfiles/video/"; // 生 成 文件 路 径 


$s_sqlstr = "select * from tb_video where id = ".$_GETT[id]; // 生 成 SQL 语句 
$s_rst = $conn->execute($s_sqlstr); 
if(!($s_rst == false))f 
/' 删 除 相应 的 图 片 文件 和 音频 、 视 频 文件 "/ 
if(unlink($file_path.$s_rst>fields[16]) and unlink($file_path.$s_rst->fields[2])){ 
/* 生 成 删除 语句 */ 
$d_sqlstr = "delete from tb_video where id = ".$_GETIid]; 
/执行 删除 语句 */ 
$d_rst = $conn->execute($d_sqlstr); 
if(!($d_rst == false)}{ 
echo "<script>alert(' 删 除 成 功 ');location='main.php?action=video';</script>"; 
Jelse{ 
echo "<script>alert(' 删 除 失败 ');history.go(-1);</script>"; 
jelse 
echo "<script>alert(' 删 除 失 败 ');history.go(-1);</script>"; 
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8.12 管理 员 设 置 模块 设计 


8.12.1 管理 员 设 置 模块 概述 


管理 员 设 置 模块 主要 包括 管理 员 信息 添加 、 管 理 员 信息 删除 及 管理 员 信 息 冻 结 3 个 部 分 。 管 理 员 
可 以 通过 后 台 管 理 功能 导航 进入 管理 员 设置 页 面 (memberphp )。 在 管理 员 设 置 页 面 中 ， 管 理 员 可 以 
进行 管理 员 信息 的 添加 操作 , 或 进行 会 员 信 息 的 冻结 或 删除 等 操作 。 管理 员 设置 用 例 图 如 图 8.35 所 示 。 


Ed 
过 加 管理 员 
管理 员 


置 ) 一 > 
~、 
删除 管理 员 


图 8.35 管理 员 设置 用 例 图 
8.12.2 ”管理 员 设置 模块 技术 分 析 


管理 员 可 以 单 击 管理 员 设 置 页 面 的 “冻结 ”或 “解冻 ”按钮 ， 对 管理 员 信息 进行 冻结 或 解冻 的 操 
作 。 当 管理 员 登 录 时 ， 首 先 查看 数据 库 中 的 whether 字段 ， 如 果 为 1， 则 说 明 该 用 户 处 于 激活 状态 ; 否 
则 ， 将 显示 提示 窗口 ， 告 诉 该 用 户 此 账号 已 被 冻结 等 提示 信息 。 

对 管理 员 信息 的 冻结 或 解冻 操作 主要 是 根据 传递 的 管理 员 的 id， 将 数据 库 中 的 whether 字段 更 新 。 
程序 代码 如 下 : 

倒 程 17 ”代码 位 置 ， 光盘 \TM\08\online\admin\m_freeze_chk.php 


<?php 
session_start(); // 开 启 session 支持 
include "conn/conn.php"; /1/ 引 入 数据 库 链 接 文件 
include "inc/chec.php"; // 引 入 权限 检查 文件 


/根据 whether 值 ， 进 行 取 反 操作 */ 
if($_POST[whether] == "1") 


$wt = "0"; 
else if($_POST[whether] == "0") 
$wt = "1"; 
else{ 
echo "<script>alert(' 非 法 操作 !");history.go(-1);</script>"; 
exit(); 


} 

/根据 管理 员 id 和 赋值 后 的 gwt， 生 成 update 语句 */ 

$o_sqlstr = "update tb_manager set whether = ".$wt." where id = ".$_POST[id]; 
/执行 update 语句 */ 

$o_rst = $conn->execute($o_sqlstr); 


_ 国 
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if(!($0_rst == false)X{ 
echo "<script>alert(' 操 作成 功 ');location='main.php?action=manager;</script>"; 


} 
?> 
该 方法 实现 很 简单 ， 但 却 十 分 实用 ， 在 实行 会 员 制 的 网 站 上 ， 在 可 以 发 帖 、 回 复 的 论坛 中 ， 都 可 
以 使 用 这 个 方法 来 管理 。 


8.12.3 ”管理 员 添 加 的 实现 过 程 


国 ”管理 员 添 加 使 用 的 数据 表 : tb_manager 

管理 员 可 以 通过 管理 员 设 置 页 面 打 开 管 理 员 信息 添加 页 面 (addmanager.php)。 在 管理 员 信 息 添加 
页 面 中 ， 管 理 员 可 以 填写 表单 并 将 表单 提交 到 处 理 页 进行 数据 处 理 ， 以 完成 管理 员 信 息 添加 的 操作 。 
管理 员 添加 页 面 的 运行 结果 如 图 8.36 所 示 。 


候 理 员 信 息 条 加 


SF: 本 


图 8.36 管理 员 添加 页 面 的 运行 结果 
管理 员 添 加 页 面 涉 及 的 HTML 表单 元 素 如 表 8.5 所 示 。 
表 8.5 管理 员 添 加 页 面 涉 及 的 HTML 表单 元 素 


名 称 类 型 重要 属性 含义 
list form method="post" action="addmanager chk.php" 表单 
names, text id="names" size="30" 管理 员 名 称 
password password id="password" size="30" 密码 

_password2 | password id="password2" size="30" 密码 确认 


<option value=" 视 频 目 录 管 理 员 " selected> 视 频 目 录 管 理 员 </option> 
<option value=" 音 频 目录 管理 员 "> 音 频 目录 管理 员 </option> 
<option value=" 视 频数 据 管理 员 "> 视 频数 据 管 理 员 </option> 


grade select <option value=" 音 频数 据 管理 员 "> 音频 数据 管理 员 </option> 管理 权限 
<option value=" 会 员 数 据 管 理 员 "> 会 员 数 据 管 理 员 </option> 
<option value=" 会 员 等 级 管理 员 "> 会 员 等 级 管理 员 </option> 
realname text id="realname" size="30" 真实 姓名 
Submit button class="submit" onClick="check():" “添加 ”按钮 
Submit2 button value=" 返 “ 回 " class="submit" onClick="javascripttop.window.closeO” “返回 ”按钮 


管理 员 信 息 添加 的 数据 处 理 主要 是 根据 提交 的 表单 数据 ， 在 数据 库 中 添加 对 应 的 记录 ， 首 先 判断 


@ 
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输入 的 名 称 是 否 存在 ， 如 果 存 在 ， 显 示 提 示 语 ， 并 返回 到 上 一 步 ， 如 果 不 存在 这 种 错误 ， 生 成 insert 
语句 ， 添 加 新 管理 员 。 程 序 代码 如 下 : 
倒 程 18 ”代码 位 置 ， 光 盘 \TM\08\online\admin\addmanager.php 


<?php 
session_start(); /开启 session 支持 
include "conn/conn.php"; /| 载 入 数据 库 链 接 文件 
include "inc/chec.php"; / 载 入 权限 检查 文件 


$a_sql="select * from tb_manager where name=".$_POSTInames].""; /检查 是 否 有 重 名 
$a_rst = $conn->execute($a_sql); 
让 如 果 管 理 员 名 称 存在 ， 弹 出 提示 框 ， 并 返回 到 上 一 步 */ 
if(!$a_rst->EOF) 
echo "<script>alert(' 该 名 称 的 管理 员 已 经 存在 ， 请 更 换 名 称 ');history.go(-1);</script>"; 
else{ 
让 生成 insert 语句 */ 
$a_sqlstr="insert into tb_manager values(",".$ POSTInames].",".$_ POST[password].",".$_POST 
[grade].",".$_POSTIrealname].",".date("Y-m-d"). ","1")"; 
让 执行 insert 语句 */ 
$a_rst1 = $conn->execute($a_sqlstr); 
让 如 果 管 理 员 添 加 成 功 */ 
if(!($a_rst1 == false)X{ 


<script> 
让 刷新 父 窗口 ， 弹 出 提示 框 ， 最 后 删除 当前 窗口 */ 
top.opener.location.reload(); 
alert(" 管 理 员 添加 成 功 "); 
top.window.close(); 
</script> 


else 
echo "<script>alert(' 添 加 失败 ".$a_sqlstr.");history.go(-1);</script>"; 


8.12.4 管理 员 删 除 的 实现 过 程 


国 管理 员 删 除 使 用 的 数据 表 : tb_manager 
管理 员 信息 删除 主要 是 根据 传递 的 管理 员 的 id， 将 数据 库 中 对 应 的 数据 删除 。 程 序 代 码 如 下 : 
倒 程 19 ”代码 位 置 ， 光盘 \TM\08\online\admin\ del_mfreeze_chk.php 


<?php 
session_start(); /开启 session 支持 
include "conn/conn.php"; // 载 入 数据 库 链 接 文 件 
include "inc/chec.php"; // 载 入 权限 检查 文件 
$d_sqlstr = "delete from tb_manager where id = ".$_GETIidj; /生成 delete 语句 
if(!($d_rst = $conn->execute($d_sqlstr)) == false) // 判 断 删除 是 否 成 功 

echo "<script>alert(' 删 除 成 功 ');location='main.php?action=manager;</script>"; 

else 
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echo "<script>alert(' 删 除 失败 ');history.go(-1);</script>"; 
Wy 


8.13 ”开发 技巧 与 难点 分 析 


在 本 系统 中 ， 多 个 地 方 使 用 了 无 边框 窗口 和 刷新 父 级 窗口 的 技术 。 下 面 来 学 习 一 下 有 关 这 两 方面 
的 知识 要 点 。 


8.13.1 无 边框 窗口 


平时 看 到 的 浏览 器 窗口 都 是 有 边框 的 ， 如 果 偶尔 有 一 个 无 边框 的 窗口 弹出 ， 屠 么 将 会 更 加 引信 注 
目 。 实 现 无 边框 窗口 使 用 的 是 JavaScript 脚本 中 的 open() 方 法 ， 该 方法 的 常用 格式 如 下 : 
window.open(urlname,features,replace); 
open() 方 法 中 的 参数 说 明 如 表 8.6 所 示 。 
表 8.6 open() 方 法 中 的 参数 说 明 


在 弹出 窗口 中 要 打开 的 地 址 


要 打开 窗口 的 名 字 。 在 使 用 target 属性 时 会 用 到 ， 可 以 为 空 
列举 的 窗口 特征 。 如 果 不 写 任何 参数 ， 那 么 默认 的 参数 多 数 情况 下 是 关闭 的 
一 个 boolean 值 ， 指 出 是 否 使 用 url 蔡 换 当前 内 容 。 该 参数 可 以 省 略 不 写 


下 面 应 用 window 对 象 的 open() 方 法 打开 一 个 指定 大 小 的 窗口 。 代 码 如 下 : 


<script language="javascript"> 
wopen = open("list.php","query","height=500,width=665,scrollbars=no,toolbar=yes ,location=yes"); 
</script> 


运行 结果 如 图 8.37 所 示 。 


8.37 ”使 用 open() 方 法 弹出 窗口 
除了 上 面 实例 中 给 出 的 几 个 窗口 特征 ， 表 8.7 还 列 出 了 其 他 几 个 常用 的 特征 参数 及 说 明 。 


第 8 章 online 影视 365 网 (Apache+PHP+ADODB+phpMyAdmin+MySQL 5.0 实现 ) 


表 8.7 浏览 器 窗口 的 外 观 样式 


属 性 值 说 了 明 
left 新 窗口 的 左 坐 标 
top | 新 窗口 的 上 坐标 ， 配 合 left 可 以 定位 一 个 窗口 的 弹出 位 置 
resizable | 是 否 可 以 通过 拖 电 来 调整 新 窗口 的 大 小 ， 默 认为 no 
status, 在 新 窗口 中 是 否 显示 状态 栏 ， 默 认为 no 


人 注意 襄 特 在 是 用 年 了 赋值 、 到 号 分 天 的 ， 在 过 号 或 等 号 前 后 不 委 加 上 空格 ， 因 为 在 有 些 济 
览 器 中 会 显示 错误 。 


8.13.2 ”刷新 父 级 窗口 和 关闭 子 窗口 


在 打开 的 子 窗口 进行 一 系列 的 操作 ， 如 添加 、 删 除数 据 后 ， 就 要 对 当前 窗口 进行 关闭 ， 最 后 返回 
父 级 窗口 中 ， 但 父 级 窗口 中 的 数据 可 能 仍然 是 没有 添加 或 删除 之 前 的 记录 数 ， 这 时 就 可 以 调用 
JavaScript 脚本 对 父 窗口 进行 刷新 了 。 这 种 对 父 级 窗口 的 刷新 只 有 对 弹出 窗口 十 有效 ， 使 用 时 要 注意 。 
该 脚本 的 使 用 格式 如 下 : 


top.opener.location.reload(); 


operner 属性 是 子 窗口 的 最 高 层 window 对 象 才 有 的 属性 ,配合 location 对 象 可 以 对 父 窗口 进行 刷新 


格式 如 下 : 


top.opener.location.href="operation.php"; 


关闭 子 窗口 使 用 如 下 代码 即 可 : 


Window.close(); 


不 过 这 里 要 注意 的 是 : close0) 方 法 在 关闭 子 窗口 时 , 不 弹出 提示 就 直接 关 掉 浏览 器 了 ,但 对 于 父 级 
窗口 ， 则 弹出 提示 框 ， 经 确认 后 才 可 以 关闭 浏览 器 。 


8.14 ADODB 连接 数据 库 技术 专题 


PHP 为 各 种 版 本 的 数据 库 定义 了 庞大 的 函数 群 来 支持 ， 而 且 支持 得 很 完整 ， 那 为 什么 本 系统 要 使 
用 ADODB 来 对 数据 库 进 行 操作 呢 ? 原因 在 于 ADODB 的 优势 : 一 次 编写 ， 不 需 修改 。 也 就 是 说 ， 当 
系统 使 用 的 数据 库 需 要 更 换 时 ， 只 要 更 新 数据 库 、 数 据 表 即 可 ， 不 需要 再 重新 编写 源码 了 ， 这 为 后 期 
维护 节约 了 一 大 笔 开 销 ， 而 且 不 需要 长 时 间 关 闭 网 站 ， 这 是 PHP 函数 所 无 法 比拟 的 。 本 章 的 技术 专题 


就 来 着 重 学 习 一 下 ADODB 技术 。 
® 
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8.14.1 ADODB 简介 


由 于 PHP 存 取 函数 没有 标准 化 ， 不 同 数 据 库 间 的 函数 名 称 、 参 数 差异 很 大 ， 在 更 换 数 据 库 时 ,会 
带 来 大 量 的 代码 修复 工作 。 这 时 就 需要 一 组 函数 库 来 隐藏 不 同 资料 库 函 数 界面 间 的 差异 ， 让 开发 者 可 
以 很 简单 地 去 切换 资料 库 , 这 就 是 ADODB。ADODB 目前 支持 MySQL、Oracle、Microsoft SQL Server、 
Sybase、PostgreSQL、Foxpro、Access、ADO 和 ODBC 等 大 多 数 知 名 的 、 不 知名 的 数据 库 类 型 。 


8.14.2 ADODB 的 安装 


要 使 用 ADODB 来 操作 数据 库 ， 首 先 就 要 获取 和 安装 ADODB。 读 者 只 要 到 网 上 下 载 ADODB 包 ， 
解压 到 Web 服务 器 目录 下 即 可 。 


入 注意 要 使 用 ADODB， 使 用 的 PHP 必须 是 4.01 以 上 的 版 本 。 


8.14.3 ADODB 常用 函数 及 说 明 


在 8.5.2 节 中 已 经 介绍 过 了 使 用 ADODB 的 一 般 步骤 , 在 实际 使 用 中 ， 需 要 根据 情况 进行 不 同 的 处 
这 时 就 需要 对 步骤 及 功能 进行 扩展 ， 下 面 就 来 介绍 儿 种 经 常会 用 到 的 函数 和 常量 。 


1. PConnect($host, $user,$password, $database) 


参数 说 明 如 下 。 

$host: 数据 库 服 务 器 地 址 。 

$user: 数据 库 用 户 名 。 

S$password: 数据 库 密码 。 

$database: 连接 的 数据 库 。 

如 果 连 接 成 功 回 传 ttue， 失 败 则 回 传 false。 

2. SelectLimit($sql,$numrows=-1,$offset=-1) 


参数 说 明 如 下 。 

$sql: 执行 的 SQL 语句 。 

S$numrows: 显示 的 记录 条 数 。 

S$offset: 从 第 几 条 记录 开始 计算 。 

该 函数 和 MySQL 中 的 limit 子 句 相似 ， 但 要 注意 ，limit 后 面 的 两 个 参数 的 顺序 是 (limit 
offsebnumrows)， 该 函数 是 完全 相反 的 。 


3. ErrorMsg() 


回 传 错误 信息 。 当 Execute() 函 数 出 现 错误 时 ， 使 用 ErrorMsg0 函 数 可 以 查看 最 后 出 错 的 相关 信息 。 
使 用 方法 如 下 : 


理 


S$recordset = $conn->Execute("select * from tb_audio") 


他 
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if(l$recordset) /判断 execute() 函 数 执行 中 是 否 有 错误 
echo $conn->ErrorMsg(); 
else{ 


即使 没有 出 现 错误 ，ErorMsg0 也 会 回 传 最 后 的 状态 值 ， 这 种 情况 下 一 般 不 需要 输出 。 
4. RecordCount() 

回 传 结果 集中 的 记录 数 。 使 用 方法 如 下 : 

eal ='select * from tb_account'; 


$rs = $conn->Execute($sql); /执行 SQL 语句 
echo $rs->RecordCount(); /输出 查询 结果 数 


5. FieldCount() 
回 传 结果 集中 的 字段 数 。 使 用 方法 同上 面 的 RecordCountO) 函 数 。 
6. MoveNext()、Move()、MoveFirst()、MoveLast() 


加 ”MoveNext(): 移动 指针 到 一 下 条 记录 。 

Move($to): 移动 指针 到 指定 的 列 〈$to)。 这 里 使 用 的 是 绝对 定位 ， 如 果 $to 是 10， 就 移动 到 
结果 集中 的 第 10 条 记录 ; 如 果 $to 大 于 记录 总 数 ， 指 针 将 移动 到 最 后 。 需 要 注意 的 是 ， 部 分 
数据 库 不 支持 向 后 移动 指针 。 

MoveFirst(): 实际 上 就 是 Move(0)， 指 针 移 动 到 结果 集 头 部 ， 部 分 数据 库 不 支持 该 函数 。 
MoveLast(): 指针 移动 到 结果 集 的 尾部 ， 也 可 以 表示 为 Move($recordCount() - 1)， 部 分 数据 库 
不 支持 此 函数 。 

7. $ADODB_FETCH_MODE 


该 常量 决定 了 结果 集 以 哪 种 方式 将 数据 回 传 。 在 有 些 数据 库 中 ， 使 用 $recordset->fields[] 仅 支持 字 
段 编号 为 索引 ， 即 0、1、2 等 ， 不 支持 字段 名 (如 id，user) 作 索 引 ， 这 时 可 以 使 用 该 常量 进行 设置 ， 
该 常量 的 值 有 : 


加 回 


加 回 


define(ADODB_FETCH_DEFAULT',0); // 系 统 默认 值 
define(ADODB_FETCH_NUM',1); // 以 字段 编号 为 索引 
define(ADODB_FETCH_ASSOC',2); // 以 字段 名 称 为 索引 
define(ADODB_FETCH_BOTH',3); // 同 时 用 时 编号 和 名 称 
这 时 上 面 的 步骤 可 以 改 为 : 


$conn = &ADONewConnection("mysq"); 
$conn->PConnect("db_online"); 
$ADODB_FETCH_MODE =ADODB_FETCH_ASSOC; 
$recordset = $conn->Execute("select * from tb_audio"); 
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if(l$recordset) // 判 断 Execute() 函 数 执行 中 是 否 有 错误 
echo $conn->ErrorMsg(); 
else{ 
while(!$recordset->EOF){ // 如 果 没 有 错误 ， 则 配合 wihle 语句 循环 输出 结果 
echo $recordset->fields[id]."<br>"; 
S$recordset->movenext(); // 指 针 下 移 
} 


这 里 建议 将 SADODB _FIELDS_MODED 的 值 固定 为 ADODB_FETCH NUM 及 ADODB_FETCH_ 
ASSOC， 不 建议 使 用 ADODB_FETCH _ BOTH， 因为 许多 驱动 程序 并 不 支持 该 常量 。 
ADODB 的 常用 函数 就 介绍 这 么 多 了 ， 如 果 读 者 感 兴趣 ， 可 以 下 载 ADODB 手册 来 学 习 更 多 内 容 。 


8.14.4 不 同 数据 库 之 间 的 转换 


使 用 ADODB 的 优势 就 是 可 以 在 以 后 的 升级 中 ， 更 换 不 同 的 数据 库 而 不 用 大 量 地 改动 源 程序 。 但 
不 同 数据 库 之 间 ， 有 的 SQL 语句 或 语法 是 不 相同 的 ， 这 时 就 要 做 一 些小 幅 的 调整 。 
(1) 在 conn.php 文件 中 ， 更 改 连接 的 数据 库 类 型 和 参数 。 


$conn = &ADONewConnection(mysql); /建立 MySQL 连接 
$conn->PConnect("localhost","root","root","db_online"); /| 连接 db_online 数据 库 


如 使 用 SQL Server 时 ， 上 面 的 链接 语句 更 改 为 : 


$conn = &ADONewConnection(mssql); 
$conn->PConnect("localhost","sa",”™,"db_online"); 


(2) 注意 包含 时 间 类 型 的 SQL 语句 。 

不 同 的 数据 库 对 于 时 间 的 表示 方法 是 不 同 的 ， 如 MySQL 使 用 时 间 类 型 时 ， 就 像 使 用 字符 类 型 一 
样 ， 放 到 单 引 号 中 就 可 以 了 ， 但 Access 则 需要 放 到 “#” 之 间 。Oracle 虽然 也 是 使 用 单 引 号 ， 但 它 的 格 
式 为 “日 -月 -年 ” 而 其 他 数据 库 大 多 都 是 “年 -月 -日 ”。 

(3) 尽量 使 用 通用 的 SQL 语句 。 

尽管 国际 标准 化 组 织 (ISO) 在 1987 年 创建 了 SQL 的 一 个 标准 ， 但 在 各 种 SQL 之 间 仍 然 存在 差 
别 。 所 以 为 了 增加 通用 性 ， 最 好 了 解 一 下 数据 库 间 的 不 同 。 例 如 ，MySQL 中 的 limit 子 句 在 Oracle 中 
就 不 被 支持 。 


online 影视 365 网 实现 了 一 个 在 线 播放 平台 。 该 网 站 具备 一 个 影视 网 站 所 需要 的 所 有 的 基本 功能 ， 
包括 影视 文件 的 上 传 、 下 载 、 播 放 及 介绍 。 还 介绍 了 JavaScript 脚本 语言 的 一 些 实用 技巧 ， 最 后 介绍 了 
本 章 的 核心 内 容 : ADODB 连接 数据 库 技 术 。 和 希望 通过 本 章 的 学 习 ， 读 者 不 仅 可 以 建立 一 个 自己 的 影 
视 网 ， 而 且 可 以 熟练 使 用 ADODB 来 连接 任意 一 个 数据 库 。 有 条 件 的 读者 可 以 试 着 将 本 系统 的 数据 库 
改 为 SQL Server 或 Oracle， 来 增加 对 ADODB 的 了 解 及 熟练 程度 。 


@ 


AS es 


Ee 时 


明日 知道 网 站 


(Zend Framework 实现 ) 


在 金 球 信息 化 高 速 发 展 的 今天 ， 互联 网 已 经 成 为 资源 共享 、 互动 
交流 、 办 公 及 娱乐 的 最 为 广泛 和 有 效 的 平台 ， 如 何 高 效 地 利用 这 些 资 
源 已 经 成 为 网 民 们 最 为 关注 的 问题 。 百 度 、 谷 歌 等 互联 网 搜索 工具 的 
出 现 已 经 在 很 大 程度 上 提高 了 信息 共享 的 进程 。 本 章 所 介绍 的 明日 知 
道 网 站 是 针对 IT 技术 问题 进行 搜索 、 为 程序 开发 人 员 提 供 交 互 的 平 
台 ， 与 互联 网 上 一 些 知 名 的 搜索 工具 相 比 更 具 专 业 化 将 点 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

MW 网 站 项 目 开 发 的 基本 流程 
Zend Framework 框架 的 搭建 方法 
MVC 开发 模式 的 应 用 
类 似 Google 搜索 条 的 制作 方法 
查询 结果 的 关键 字 描 红 输 出 
Mm 在线 编辑 中 的 原理 及 制作 方法 


吾 吾 吾 至 
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9.1 开发 背景 


X X 科 技 有 限 公 司 是 一 家 以 IT 教育 、 互 联网 开发 和 运营 为 主 的 高 科技 互联 网 企业 。 在 多 年 的 经 营 
中 积累 了 大 量 的 IT 技术 方面 的 资源 ， 为 了 提升 公司 的 知名 度 ， 使 资源 得 以 共享 ， 并 创造 更 大 的 广告 利 
润 , 公司 组 织 了 一 支 团队 来 开发 明日 知道 网 站 , 争取 打造 一 个 国内 知名 的 专业 IT 技术 搜索 和 交互 平台 。 

明日 知道 网 站 是 为 了 使 搜索 变 得 更 加 专业 ， 以 方便 广大 程序 相关 工作 者 而 打造 的 专业 化 互联 网 搜 
索引 擎 。 通 过 该 搜索 引擎 ， 程 序 开发 人 员 可 以 方便 地 查找 到 有 关 软 件 开发 方面 的 技术 问题 和 文章 。 并 
且 明日 知道 网 站 提供 了 互动 平台 ， 在 学 习 编程 过 程 中 如 果 遇 到 了 技术 问题 ， 可 以 将 问题 发 表 于 明日 知 
道 网 站 上 ， 其 他 注册 会 员 可 以 对 提出 的 问题 进行 解答 ， 这 样 即便 通过 明日 知道 没有 搜索 到 要 查找 的 内 
容 ， 也 可 以 通过 互动 方式 将 问题 解决 。 专 业 化 搜索 工具 是 有 一 定 发 展 潜能 的 项 目 课题 。 


9.2 需求 分 析 


对 于 明日 知道 这 类 信息 检索 类 网 站 来 说 ， 功 能 强大 的 搜索 模块 显得 尤为 重要 ， 借 鉴 其 他 知名 的 搜 
索 工 具 ， 方 便 、 高 效 的 搜索 引擎 应 该 具备 关键 字 提示 列表 、 分 词 查询 和 查询 结果 的 描 红 输出 等 功能 。 
为 了 避免 用 户 要 查找 的 内 容 在 本 站 没有 查找 到 结果 ， 所 以 还 需要 增加 用 户 交 流 模块 ， 这 样 即 使 用 户 的 
问题 在 本 站 没有 查找 到 答案 ， 也 可 以 将 问题 发 布 出 去 ， 其 他 用 户 就 可 以 对 该 问题 进行 回答 ， 直 到 解决 
为 止 。 所 以 明日 知道 网 站 应 该 打造 一 个 服务 开发 人 员 ， 对 资源 进行 不 断 积累 的 综合 平台 。 

通过 实际 调查 ， 要 求 明日 知道 网 站 应 该 具备 以 下 功能 : 

类 似 百度 和 Google 等 搜索 引擎 的 搜索 条 。 
多 关键 字 的 分 词 查 询 。 

查询 结果 的 关键 字 描 红 输 出 。 

使 用 在 线 编辑 器 发 布 和 回复 问题 。 
问题 的 引用 。 


加 


加 


加 图 回 


9.3 功能 结构 


“ 磨 刀 不 误 砍 柴 工 ”， 建 立 一 个 完善 的 
系统 功能 图 对 项 目的 整体 开发 具有 指导 意 
义 ， 因 此 在 项 目 编码 前 应 给 出 项 目前 台 和 后 
台 的 系统 功能 结构 图 。 其 中 ， 明 日 知道 网 站 
前 台 功 能 结构 图 如 图 9.1 所 示 ， 后 台 功 能 结 
构图 如 图 9.2 所 示 。 


图 9.1 网 站 前 台 功能 结构 图 。 图 9.2 网 站 后 台 功 能 结构 图 
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9.4 


功 能 


明日 知道 网 站 ( Zend Framework 实现 ) 信 令 怒 


Le 


人 见 


明日 知道 网 站 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 几 个 


典型 功能 的 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 

明日 知道 首页 由 网 站 Logo、 语 言 导航 条 、 搜 索 
区 、 友 情 链 接 和 版 权 声明 等 部 分 组 成 。 页 面 简洁 大 
方 、 突 出 核心 ,便于 用 户 使 用 ， 其 运行 效果 如 图 9.3 
所 示 。 

会 员 登录 模块 完成 注册 会 员 的 登录 操作 ， 并 且 
融入 验证 码 技术 ， 同 时 加 入 了 会 员 注 册 的 超 链接 ， 
其 运行 效果 如 图 9.4 所 示 。 

问题 分 类 模块 展示 明日 知道 网 站 中 专题 技术 的 


理 | 下 


Ol 


他 本 
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图 9.3 明日 知道 首页 


分 类 ， 同 时 展示 每 类 中 包含 的 问题 和 回复 情况 ， 其 运行 效果 如 图 9.5 所 示 。 
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图 9.5 问题 分 类 页 面 


输出 查询 结果 模块 展示 搜查 的 结果 ， 并 且 对 查询 的 关键 字 进 行 描 红 ， 同 时 对 查询 结果 进行 分 页 输 


出 ， 其 运行 效果 如 图 9.6 所 示 。 
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图 9.6 查询 结果 的 分 页 输出 页 面 
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9.5 数据 库 设 计 


在 进行 项 目 开发 前 需要 根据 实体 属性 建立 数据 库 实体 模型 ， 然 后 根据 实体 之 间 存在 的 客观 联系 设 
计 出 合理 的 数据 表 。 由 于 数据 库 是 项 目 内 容 的 主要 载体 ， 所 以 设计 出 合理 的 数据 库 、 表 及 表 间 关系 显 
得 非常 重要 。 


9.5.1 数据 库 分 析 


明日 知道 网 站 采用 PHP 语言 开发 ， 并 且 该 项 目 为 资源 搜索 型 网 站 ， 数 据 量 会 很 大 ， 所 以 数据 库 稳 
定 、 高 效 的 存储 和 检索 能 力 显得 非常 重要 ,综合 上 述 因 素 及 PHP 的 完全 跨 平台 等 特性 ,应 该 首选 MySQL 
数据 库 作为 明日 知道 网 站 数据 的 载体 。 在 长 时 间 的 工作 中 也 总 结 出 , PHP 和 MySQL 是 最 佳 的 黄金 搭档 。 


9.5.2 数据库 概 念 设计 

根据 明日 知道 网 站 的 需求 分 析 、 功 能 结构 分 析 ， 规 划 出 整个 项 目的 数据 库 实 体 应 该 包括 问题 类 别 
实体 、 关 键 字 实体 、 问 题 实 体 、 回 复 实体 和 用 户 实体 等 部 分 。 下 面 介绍 这 几 个 实体 的 E-R 图 。 

1， 问 题 类 别 实体 

明日 知道 网 站 定位 打造 一 个 专业 、 高 效 的 在 线 搜索 工具 ， 这 样 就 需要 搜索 内 容 按 类 别 划分 。 资 源 
分 类 明确 ， 可 以 有 效 地 提高 用 户 检索 的 效率 ， 方 便 用 户 定位 查找 。 问 题 类 别 实体 包括 类 别 ID、 类 别名 
称 、 类 别 描述 和 添加 时 间 等 属性 ， 其 实体 E-R 图 如 图 9.7 所 示 。 


2. 问题 实体 


问题 实体 是 对 用 户 提出 问题 的 抽象 ， 根 据 功 能 分 析 ， 问 题 实体 包括 问题 ID 号 、 主 题 、 内 容 、 添 加 
时 间 、 置 顶 标识 、 用 户 ID、 类 别 ID、 浏 览 次 数 、 上 传 文件 名 和 精华 问题 标识 等 属性 ， 其 实体 E-R 图 如 
图 9.8 所 示 。 


CK 数 


类 别名 称 类 别 描述 
《类 别 ID > 问题 类 别 添加 时 间 


9.7 问题 类 别 实体 E-R 图 9.8 问题 实体 E-R 图 
3. 回复 实体 


回复 实体 包括 回复 ID 号、 主题、 内容、 最 佳 答案 标识 、 添 加 时 间 、 用 户 ID、 主 题 ID 和 类 别 ID 
等 属性 ， 其 实体 E-R 图 如 图 9.9 所 示 。 
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4. 关键 字 实体 
关键 字 实 体 是 对 用 户 输 入 关键 字 进 行 记录 的 抽象 ， 该 实体 包括 ID 号 、 关 键 字 和 检索 次 数 等 属性 ， 
其 实体 E-R 图 如 图 9.10 所 示 。 
C0 


图 9.9 
5. 用 户 实体 


J 
EE 问题 回复 | CC 县 佳 答案 标识 > 
CH Cm CG 他 关键 字 记录 实体 | 搜索 次 数 “> 


回复 实体 E-R 图 图 9.10 关键 字 实 体 E-R 图 


用 户 实体 是 对 网 站 注册 用 户 的 抽象 ， 其 属性 包括 用 户 ID、 用 户 昵称 、 登 录 密码 、E-mail、 联 系 电 
话 、 所 在 地 、 注 册 时 间 、 头 像 名 称 、 发 帖 次 数 、 回 帖 次 数 和 


用 户 类 别 标识 等 属性 ， 其 实体 E-R 图 如 图 9.11 所 示 。 
9.5.3 创建 数据 库 及 数据 表 
根据 需求 分 析 及 数据 库 概念 设计 ， 可 以 制定 出 明日 知 


道 网 站 数据 库 〈 这 里 命名 为 db_known) 应 该 包括 问题 类 别 
表 (tb_bbstype) 、 问 题 表 (tb_title) 、 回 复 表 (tb_reply) 、 


关键 字 表 (tb_keywords) 和 用 户 表 (tb_user) 等 5 个 表 ， 图 911 用 户 实体 ER 图 
其 中 各 表 的 主要 参数 如 图 9.12 所 示 。 
下 面 将 具体 介绍 明日 知道 数据 库 中 各 个 数据 表 字 段 的 具体 描述 。 


1. 问题 类 别 
问题 类 别 表 
表 


2. 问题 表 ( 


表 (tb_bbstype) 


(这 里 命名 为 tb_bbstype) 用 于 存储 问题 类 别 ， 其 结构 如 图 9.13 所 示 。 
一 一 一 字段 类 型 整理 额外 
i int(11) auto_increment 


ut8_unicode_ci 
ut8_unicode_cl 


typename varchar(50) 。 utB_unicode_ci 


addtme datetime 
description varchar(255) utB_unicode_ci 


图 9.12 明日 知道 网 站 数据 表 主 要 参数 9.13 ”问题 类 别 表 结 构 


utB_unicode_ei 
ut8_unicode_cl 


tb title) 


问题 表 〈 这 里 命名 为 tb_title) 用 于 存储 问题 内 容 ， 其 结构 如 图 9.14 所 示 。 


3. 回复 表 《 
回复 表 (这 


tb_reply) 
有 命名 为 tb_reply) 用 于 存储 用 户 对 问题 的 回复 ， 其 结构 如 图 9.15 所 示 。 
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图 9.14 问题 表 结 构 图 9.15 回复 表 结构 


4. 关键 字 表 (tb_keywords) 

关键 字 表 (这 里 命名 为 tb_keywords) 用 于 存储 用 户 曾 搜索 的 关键 字 及 每 个 关键 字 被 搜索 的 次 数 ， 
其 结构 图 如 图 9.16 所 示 。 

5. 用 户 表 (tb_user) 

用 户 表 ( 这 里 命名 为 tb_user) 用 于 存储 用 户 信息 ， 其 结构 图 如 图 9.17 所 示 。 
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图 9.16 关键 字 表 结构 图 9.17 用 户 表 结 构 


9.6 技术 攻关 


明日 知道 网 站 在 开发 过 程 中 ， 多 处 使 用 了 较 新 的 技术 。 为 了 便于 对 该 项 目 各 个 模块 的 理解 ， 本 节 
首先 对 明日 知道 网 站 所 使 用 的 关键 技术 进行 讲解 。 


9.6.1 类似 Google 搜索 引擎 的 搜索 条 


为 了 方便 网 站 浏览 者 检索 网 站 内 容 ， 很 多 知名 网 站 的 查询 关键 字 文 本 框 都 提供 关键 字 提示 下 拉 列 
表 框 ， 这 样 用 户 就 可 以 根据 以 往 用 户 所 输入 的 关键 字 更 加 准确 地 定制 自己 的 关键 字 ， 从 而 在 很 大 程度 
上 提高 了 要 查询 内 容 被 高 效 、 准 确 查询 出 来 的 几率 。 同 样 ， 在 制作 明日 知道 网 站 的 搜索 模块 也 引用 了 
有 关键 字 提示 的 搜索 条 ， 如 图 9.18 所 示 。 

那么 ,类 似 Google 的 搜索 条 是 如 何 实现 的 呢 ? 是 不 是 特别 复杂 呢 ? 继续 向 下 学 习 将 会 寻求 到 答案 。 
在 具体 讲解 实现 过 程 前 ， 首 先 讲解 制作 该 搜索 条 的 思路 ， 当 用 户 在 关键 字 文本 框 中 输入 内 容 时 ， 通 过 
文本 框 的 onkeyup 事件 调用 来 显示 关键 字 下 拉 列 表 的 方法 ， 该 方法 通过 Ajax 技术 (这 里 使 用 Ajax 的 
框架 jQuery) ， 应 用 GET 方法 向 服务 器 发 送 请 求 ， 请 求 过 程 中 将 用 户 输入 的 关键 字 发 送 给 服务 器 ， 服 


@ 
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务 器 经 过 处 理 返 回 所 有 以 该 关键 字 开 头 的 用 户 曾 搜索 的 关键 字 及 搜索 次 数 和 与 该 关键 字 匹 配 的 结果 个 
数 。 下 面具 体 讲解 类 似 Google 搜索 条 的 实现 方法 。 


| VisualBasie Visualc+t+ ~ Ja cs ASPNET PHP 


图 9.18 类似 Google 搜索 引擎 的 搜索 条 


(1) 建立 用 于 输入 关键 字 的 文本 框 ， 这 里 使 用 Zend Framework 框架 的 formText() 视 图 助手 建立 。 
Zend Framework 框架 的 视图 助手 ， 大 部 分 用 来 生成 组 件 的 和 有 自动 转 义 变量 的 功能 。 另 外 ， 有 些 助手 
用 来 创建 基于 路 由 的 URL 和 HTML 列表 以 及 声明 变量 。 这 里 所 使 用 的 formText() 助 手 用 来 生成 HTML 
的 文本 框 组 件 。 
formText() 视 图 助手 的 语法 格式 如 下 : 


formText(string name, string value, array attribs) 
参数 说 明 如 表 9.1 所 示 。 
表 9.1 formText() 视 图 助手 的 参数 说 明 


参数 说 了 明 
name 必要 参数 ， 指 定 所 生成 文本 框 组 件 的 名 称 
value 可 选 参数 ， 文 本 框 的 内 容 ， 默 认为 null 
attribs, 


可 选 参数 ， 以 数组 形式 指定 文本 框 的 多 个 属性 ， 默 认为 null 


明日 知道 网 站 使 用 formTextO 视 图 助手 生成 搜索 文本 框 的 代码 如 下 : 
倒 程 01 ”代码 位 置 : 光盘 \TM\09\known\application\modules\default\views\scripts\index\index.phtml 
echo $this->formText(keywords', ", array('style'’=>'width:350px; height:26px; line-height:26px; font-size:15px; 
color#333333; border:0px;"，'onkeyup'=>'showKeywordsList()，'onkeydown' =>'selectList()，'autocomplete'=> 
"off)) // 通 过 formText() 视 图 助手 输出 查询 关键 字 录入 文本 框 
上 述 代 码 中 ， 将 关键 字 文 本 框 命名 为 keywords， 默 认 值 为 空 串 ， 在 属性 数组 中 ， 通 过 键 名 为 style 
的 数组 元 素 指定 文本 框 样式 ， 通 过 键 名 为 onkeyup 的 数组 元 素 指定 当 按 键 抬 起 时 所 触发 的 事件 ， 通 过 
键 名 为 onkeydown 的 数组 元 素 指 定 当 按键 按 下 时 所 触发 的 事件 ， 最 后 为 键 名 为 autocomplete 的 数组 元 
素 指定 off 值 来 去 除 文本 框 自身 的 下 拉 列 表 提示 。 
(2) 当 用 户 在 文本 框 中 输入 关键 字 时 ， 将 调用 自 定 义 的 showKeywordsList() 方 法 显示 关键 字 下 拉 
列表 ， 代 码 如 下 : 


倒 程 02 ”代码 位 置 ， 光盘 \TM\09\known\application\modules\default\views\scripts\index\index.phtml 


var index = 0; /| 关键 字 列表 项 索引 
var count = 0; /| 关键 字 列表 总 数 
function showKeywordsList(){ /显示 关键 字 下 拉 列 表 


@ 
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var keycode = event.keyCode; /用 户 按键 的 ASCII 码 
if(keycode != 40 && keycode != 38){ /| 如 果 不 是 向 上 和 向 下 方向 键 
index = 0; /关键 字 索 引 归 0 


if($(#keywords").val() == ”|| $.tim($("#keywords").val()) == ” 


$("#searchLayer").css("display", "none"); 。 // 如 果 用 户 输入 空 字符 或 空格 ， 则 隐藏 下 拉 列 表 
}else{ 
setTimeout("reSearch()", 150); /| 否则 每 隔 150 毫秒 调用 一 次 reSearch() 方 法 
} 
+ 
} 


Ah 代码 贴 十 
@ 这 里 之 所 以 要 间隔 150 毫秒 后 调用 reSearch() 方 法 ， 是 为 了 降低 连续 对 服务 器 请 求 而 带 来 的 负载 。 
上 述 代码 中 ， 首 先 使 用 JavaScript 的 event 对 象 的 keyCode 属性 获取 用 户 输入 字符 的 ASCII 码 ， 然 
后 判断 该 ASCII 码 是 否 为 40 代表 向 上 方向 键 ) 或 38〈 代 表 向 下 方向 键 》， 如 果 不 是 这 两 个 方向 键 ， 
并 且 用 户 输入 的 字符 不 为 空 串 或 空格 ， 则 间隔 150 毫秒 调用 reSearch() 方 法 来 显示 与 用 户 输入 关键 字 所 
匹配 的 关键 字 搜 索 记录 。 其 中 reSearch() 方 法 的 代码 如 下 : 
倒 程 03 ”代码 位 置 ， 光盘 \TM\09\known\application\modules\default\views\scripts\index\index.phtml 


function reSearch(){ 
if($("#keywords").val() != ”多 // 如 果 用 户 输入 的 关键 字 不 为 空 
@ $.get('<?php echo Sthis->baseUrl(/index/keywords-liskeyword/"+encodeURI($("#keywords"). val())+")?>", 


null function(dataX{ 
// 通 过 jQuery 向 服务 器 发 送 GET 请 求 


if($.trim(data) == "YX // 如 果 返 回 数据 为 空 ， 则 隐藏 下 拉 列 表 
$("#searchLayer").css("display", "none”); 
}else{ 
$("#searchLayer").html(data); // 如 果 返 回 数据 不 为 空 ， 则 显示 下 拉 列 表 
$("#searchLayer").css("display", "block"); 
@ count = parselnt($("#totalList").val()); // 获 取 列 表 中 关键 字 个 数 
} 
»); 
} 
} 
< 代码 贴 二 


@ encodeURI( 方 法 : 用 来 对 关键 字 进 行 URL 编码 。 
@ parseInt() 方 法 : 用 来 将 字符 串 转换 为 整 型 。 


上 述 代码 中 ， 通 过 jQuery 的 $.get0 方 法 向 服务 器 发 送 请 求 ， 如 果 返 回 结果 不 为 空 串 ， 则 将 内 容 显 
示 在 下 拉 列 表 中 。 


/ 
B 
Mt 通过 Ajax 技术 向 服务 器 发 送 请 求 ， 传 递 的 内 容 应 该 为 utf8 编码 ， 如 果 不 是 ， 应 该 使 用 函 
数 iconv0 对 传递 的 内 容 进 行 转 码 。 


(3) 当 用 户 移动 键盘 的 上 下 方向 键 时 ， 列 表 中 所 选中 的 项 以 特殊 色 显 示 ， 并 在 关键 字 文 本 框 中 显 
示 当 前 选中 项 内 容 。 代 码 如 下 : 


@ 
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倒 程 04 ”代码 位 置 : 光盘 \TM\09\known\application\modules\default\views\scripts\index\index.phtml 


function selectList(){ 
var keycode = event.keyCode; // 获 取 用 户 输入 字符 的 ASCII 码 
if(keycode == 40X{ // 如 果 用 户 按 向 下 方向 键 
$("#listltem_"+index).css("background-color", #158CD0"); 。“”// 更 改 当前 所 选项 的 背景 色 
$("#listltem_"+index).css("color", "#FFFFFF"); // 更 改 当前 所 选项 的 前 景 
$("#keywords").val($.tim($("#istltem_li_"+index).html())); // 将 当前 所 选项 显示 在 文本 框 中 
iflindex > OX 


S$("#listltem_"+parselnt(index-1)).css("background-color", "#FFFFFF"); 
// 更 改 当前 项 前 一 项 的 背景 色 
$(" 夫 istltem_"+parselnt(index-1)).css("color", "#333333"); /更 改 当前 项 前 一 项 的 前 景色 


} 
if(index < count-1){ // 如 果 当 前 索引 小 于 总 关键 字数 减 1 
index++; // 索 引 数 增 1 
} 
Jelse if(keycode == 38X{ // 如 果 用 户 按 向 上 方向 键 
if(index > OX // 如 果 索 引 大 于 0 


S$("#listltem_"+parselnt(index-1)).css("background-color", "#158CDO"); 
// 更 改 当前 项 前 一 项 背景 色 
$("#listltem_"+parselnt(index-1)).css("color", "#FFFFFF"); /更 改 当前 项 前 一 项 前 景色 
$("#keywords").val($.trim($("#istltem_li_"+parselnt(index-1)).html())); 
// 将 当前 所 选项 显示 在 文本 框 中 


} 
$("#istltem_"+index).css("background-color", "#FFFFFF"); 
/更 改 当前 项 背景 色 
$("#istltem_"+index).css("color", "#333333"); // 更 改 当前 项 前 景色 
iflindex > 1){ // 如 果 索 引 大 于 1 
index—; 1/ 索引 数 减 1 
半 
} 


3 


上 述 代 码 首先 获取 用 户 输入 字符 的 ASCII 码 ， 然 后 判断 该 值 是 否 为 向 上 方向 键 或 向 下 方向 键 所 对 
应 的 ASCI 码 ， 如 果 是 则 通过 jQuery 的 css() 方 法 改变 所 选项 的 前 景色 和 背景 色 ， 从 而 达到 关键 字 列表 
中 内 容 上 下 切换 的 效果 。 
(4) 在 上 面 的 讲解 中 ， 已 经 提 到 客户 端 通过 GET 方法 向 服务 器 发 出 请 求 ， 那 么 发 出 请 求 后 做 了 
哪些 处 理 呢 ? 从 下 述 代 码 中 将 会 找到 答案 。 
倒 程 05 ”代码 位 置 : 光盘 \TM\09\known\application\modules\default\controllers\IndexController.php 
public function keywordsListAction () 


‘ 
@ header('content-type:text/html; charset=utf-8"); // 设 置 页 面 编码 为 utf-8 

$keyword = urldecode($this->_request->getParam('keyword')); /获取 用 户 输入 的 关键 字 并 进行 URL 解码 
@ $keywordses = $this-> keywordsModel->findByLike($keyword, 10, true); /调用 关键 字模 型 的 findByLink() 
方法 查询 所 有 以 该 关键 字 开头 的 内 容 


S$total = 0; /保存 总 匹配 关键 字数 
if ($keywordses != null) { 

S$total = count($keywordses); // 统 计 总 匹配 关键 字数 
} 


@ 
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$str = '<input type="hidden" name="totalList" id="totalList" value=”. $total .™ />"; 
foreach ($keywordses as $k => $key) { // 通 过 循环 形成 下 拉 列 表 
Sstr .= '<div id="listltem_' . $k . ”style="width:98%; clearboth; background-color#FFFFFF; cursor:pointer;" 
‘onmouseover="this.style.backgroundColor=\'#158CDO\'; this.style.color=\#FFFFFF\" onmouseout= "this.style. 
backgroundColor=\'#FFFFFR\'; this. style.color=\#333333\" onclick="setText(\" . $key['keyword] . \)">'; 
$str .= '<ul>'; 
Sstr .= ' <li id="listltem li . $k . ”style="width:60%; height:18px; line-height:18px; text-align:left; floatleft">' . 
$key['keyword’] . '</li>"; 
$str .= ' <li style="Width:38%; height:18px; line-height:18px; text-align:right; floatright;">' . $key[ searchtime] . "</li>"; 


$str .= "</ul>"; 
$str .= '</div>"; 
上 
echo $str; 
S$this->_helper->layout->disableLayout(); // 取 消 页 面 布 局 
S$this->_helper->viewRenderer->setNoRender(); /取消 视图 
< 代码 贴 二 


@header() 函 数 : 用 于 向 客户 端 浏 览 器 发 送 HTTP 头 信息 ， 这 里 用 来 指定 页 面 所 使 用 的 编码 为 utf-8。 
@ 这 里 调用 关键 字模 型 的 findByLike() 方 法 ， 获 取 关 键 字 被 搜索 最 多 的 10 条 记录 。 


上 述 代码 为 default 模块 中 IndexController 控制 器 的 keywordsListAction 动作 ， 主 要 用 来 获取 下 拉 
列表 的 数据 并 发 送 给 客户 端 。 

由 于 上 述 代码 只 需要 显示 下 拉 列 表 内 容 ， 不 需要 显示 网 站 头 和 尾 信息 ， 即 不 需要 使 用 布局 ， 所 以 
使 用 如 下 代码 取消 该 动作 的 页 面 布局 。 


$this->_helper->layout->disableLayout(); 


和 


Zend Framework 中 , 使 用 当前 控制 器 对 象 的 _helper 属性 的 layout 属性 的 disableLayout0 方 法 取消 布局 。 
同样 ， 由 于 不 需要 视图 , 直接 通过 echo 语句 就 可 以 输出 列表 中 的 内 容 ， 所 以 可 以 将 该 Action 所 使 
用 的 视图 通过 如 下 代码 取消 。 


$this->_helper->viewRenderer->setNoRender(); 


4 注意 由 于 明日 知道 网 站 使 用 Zend Framework 框架 开发 ， 学 习 本 章 内 容 需要 有 一 定 MVC 开发 
思想 和 Zend Framework 框架 基础 。 


9.6.2 ”使 用 空格 分 隔 多 关键 字 


用 户 在 使 用 百度 、Google 等 互联 网 搜索 工具 时 ， 如 果 对 多 个 关键 字 同 时 进行 搜索 ， 可 以 将 多 个 关 
键 字 用 空格 分 隔 ， 从 而 提高 查找 效率 。 
明日 知道 网 站 中 为 了 方便 用 户 查 找 ， 采 用 分 词 技术 可 以 同时 查找 多 个 关键 字 ， 例 如 用 户 输入 “ 编 


@ 
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程 ”和 “软件 ”两 个 关键 字 ， 并 且 关 键 字 用 空格 分 隔 ， 提 交 搜 索 后 将 查找 出 所 有 包含 这 两 个 关键 字 的 
内 容 ， 如 图 9.19 所 示 。 


9.19 ”多 关键 字 同 时 查询 


下 面 就 来 一 起 探讨 这 种 实用 的 分 词 技术 的 实现 方法 。 
(1) 当 用 户 输入 完 关键 字 提 交 查 询 内 容 后 ， 首 先 在 IndexController 控制 器 的 searchAction 动作 方 
法 中 使 用 getParam() 方 法 获取 用 户 提交 关键 字 的 值 ， 代 码 如 下 : 


$keywordsStr = urldecode(S$this->_request->getParam(keywords')); 


Zend Framework 框架 中 接收 页 面 间 传 递 参数 的 方法 有 3 种 ， 分 别 为 当前 控制 器 对 象 _request 属性 
的 getQuery() 方 法 、getPost() 方 法 和 getParam() 方 法 。 这 3 个 方法 的 区 别 是 : getQuery() 方 法 用 来 获取 
GET 方法 提交 的 数据 ，getPost() 方 法 用 来 获取 POST 方法 提交 的 数据 ， 而 通过 getParam() 方 法 既 可 以 获 
取 GET 方法 提交 的 数据 ， 还 可 以 获取 POST 方法 提交 的 数据 。 
(2) 在 问题 表 (tb_title) 模型 Model_DbTable_Title 中 建立 fndByLike() 方 法 ， 用 来 对 用 户 输入 的 
关键 字 进 行 模糊 查找 。 代 码 如 下 : 
倒 程 06 ”代码 位 置 ， 光盘 \TM\09\known\application\models\DbTable\Title.php 


public function findByLike ($keywordsStr, $bbstype_id = 0, $orderFlag = null, $page = 1, $pageSize = 10, 
$pageRange = 5) 
{ 


@ $arayKeyWords = explode(… $keywordsStr); /将 多 个 关键 字 用 空格 分 隔 保存 到 数组 中 
if ($orderFlag == null || $orderFlag == 0) { /指定 排序 方式 
$order = 'addtime desc'; /时 间 降 序 
} else{ 
$order = 'browse desc'; /时间 升 序 
$select = $this->getAdapter()->select(); /生成 Zend_Select 对 象 
$select->from($this->_name); /为 Zend_Select 对 象 指定 表 名 
foreach ($arrayKeyWords as $keywords) { // 人 遍历 所 有 被 空格 分 隔 的 关键 字 
if (trim($keywords) (= ") { // 如 果 关 键 字 不 为 空 串 


@$select->orWhere($this->getAdapter()->quotelnto('title like ?", '%' . $keywords . '%")); 
// 对 主题 进行 匹配 查询 
if ($bbstype_id (= 0) { 
$select->where('bbstype_id = '. $bbstype_id); 。“// 指 定 查 询 类 别 


$select->orWhere($this->getAdapter()->quotelnto(unhtmlcontent like ?', '%' . $keywords . '%')); 


// 对 内 容 进行 匹配 查询 
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if ($bbstype_id != 0){ 
$select->where('bbstype_id = '. $bbstype_id); 。“// 指 定 查 询 类 别 


上 
这 
$select->order($order); // 指 定 查 找 结 果 排序 方式 
$adapter = new Zend_Paginator_Adapter_DbSelect($select); /指定 分 页 适配器 
$paginator = new Zend_Paginator($adapter); /| 生成 Zend_Paginator 分 页 对 象 
S$paginator->setltemCountPerPage($pageSize)->setCurrentPageNumber($page)->setPageRange($pageRange); 
// 指 定 分 页 参数 
return $paginator; // 将 Zend_Paginator 对 象 作为 结果 返回 
} 
< 人 代码 贴 二 


@ explode() 函 数 : 用 来 将 字符 囊 用 指定 子囊 分 隔 ， 并 保存 到 数组 中 。 
@ $this->getAdapter()->quoteInto() 方 法 : 用 来 防止 非法 字符 注入 。 
(3) 在 searchAction 动作 方法 中 ， 通 过 tb_title 表 模 型 实例 的 对 象 调用 findByLike() 方 法 来 获取 
Zend_Paginator 对 象 对 查询 结果 进行 分 页 显示 。 代 码 如 下 : 


$paginator = $this->_ titleModel->findByLike($keywordsStr, $typeid, null, $page, $pageSize); 


(4) 在 searchAction 动作 的 视图 中 ， 通 过 foreach 循环 语句 输出 查询 结果 ， 代 码 如 下 : 


<?php foreach ($this->paginator as $search):?> 
<! 一 -省 略 HTML 代码 ， 用 来 显示 查询 结果 --> 
<?php endforeach;?> 


(5) 为 了 将 查询 结果 进行 分 页 显示 ， 还 需要 建立 分 页 导航 。Zend_Paginator 的 分 页 控制 器 对 象 的 
视图 提供 了 多 个 便于 开发 人 员 建 立 分 页 的 属性 ， 如 表 9.2 所 示 。 
表 9.2 Zend_Paginator 分 页 控制 器 对 象 的 视图 常用 属性 


属 性 说 了 明 
first 首页 页 码 号 
firstItemNumber, 当前 页 第 一 条 记录 绝对 编号 
firstPageInRange 分 页 导航 中 首页 页 码 号 
current 当前 页 的 页 号 
currentItemCount 当前 页 记录 总 数 
itemCountPerPage 每 页 显示 的 记录 数 
last 尾 页 页 码 号 
lastItemNumber 最 后 一 条 记录 号 
lastPageInRange 分 页 导航 中 最 后 一 页 页 码 号 
next 下 一 页 页 码 号 
_pageCount 总 页 数 
_pagesInRange 根据 所 使 用 导航 条 样式 ， 返 回 页 码 数组 
_Previous 上 一 页 页 码 号 
totalltemCount 记录 总 数 


@ 
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了 解 分 页 控制 器 视图 的 主要 属性 后 ， 下 面 开始 讲解 建立 分 页 导航 的 步骤 。 首 先 需要 在 search.phtml 
视图 文件 中 所 要 显示 视图 导航 的 位 置 通过 视图 对 象 的 paginationControl() 方 法 指定 分 页 导航 的 属性 ， 代 
码 如 下 : 


<?php echo $this->paginationControl($this->paginator, 'Sliding' ,searchPaginationControl.phtml', array(keywords' => 
urlencode(S$this->keywordsStr)))?> 


视图 对 象 的 paginationControl() 方 法 的 语法 格式 如 下 : 


paginationControl(Zend_Paginator paginator, string style, string paginationController, array params) 


参数 说 明 如 表 9.3 所 示 。 
表 9.3 视图 对 象 的 paginationControl() 方 法 的 参数 说 明 
说 了 明 
必要 参数 ， 指 定 Zend_Paginator 对 象 


可 选 参数 ， 指 定 导航 条 样式 ， 具 体 样式 说 明 请 参见 表 9.4 
可 选 参数 ， 指 定 分 页 控制 器 视图 文件 名 
以 数组 形式 指定 要 传递 给 分 页 控制 器 视图 的 变量 名 称 和 值 


表 9.4 导航 条 样式 说 明 


类 似 Yahoo 所 使 用 的 导航 条 


最 后 还 需要 建立 导航 视图 ， 明 日 知道 网 站 中 导航 视图 代码 如 下 : 
倒 释 07 代码 位 置 ， 光盘 \TM\09\known\application\modules\default\views\scripts\searchPaginationControl.phtml 


<ul> 
<li class="li1"> 每 页 &nbsp;<?php echo $this->itemCountPerPage?>&nbsp; 条 / 共 查 找到 &nbsp;<?php echo 
$this->totalltemCount?>&nbsp; 条 </li> 
<li class="li_cell"></li> 
<?php if($this->totalltemCount>0):?> 
<?php foreach ($this->pagesInRange as $page):?> 
<li onclick="javascript:window.location.href='"<?php echo $this->baseUrl(/index/search/keywords/'. $this-> 
keywords.'/page/'.$page)?>'"," class="<?php if($this->current==$page):?>li2_current<?php else:?>li2<?php 
endif;?>"><a href="<?php echo S$this->baseUrl(/index/search/keywords/".$this->keywords.'/page/".$page)?>" 
class="<?php if($this->current==$page):?>a2<?php else:?>a1<?php endif;?>"><?php echo $page?></a></li> 
<liclass="li_cell"></li> 
<?php endforeach;?> 
<?php if($this->last > 5):?> 
<li onclick="javascriptwindow .location.href='<?php echo S$this->baseUrl(/index/search/keywords/'. $this-> 
keywords.'"/page/".$this->last)?>";" class="li3"><a href="<?php echo S$this->baseUrl(/index/search/keywords/". 
S$this-> keywords./ page/".$this->last)?>"class="a1">...<?php echo S$this->last?></a></li> 
<?php endif;?> 
<?php endif;?> 
</ul> 


_ 国 
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用 户 使 用 明日 知道 网 站 查询 内 容 ， 将 在 查询 结果 页 面 显 示 如 图 9.20 所 示 的 分 页 导航 条 。 


9.6.3 ”关键 字 描 红 nn | 轨 [2] [3][4] [5 
图 9.20 多 关键 字 同 时 查询 


使 用 互联 网 搜索 引擎 时 ， 当 用 户 在 搜索 条 中 输入 完 关键 字 
并 提交 查询 信息 后 ， 会 在 查询 结果 显示 页 面 查 看 到 所 有 输入 的 查询 关键 字 都 以 红色 显示 ， 这 实质 就 是 
关键 字 的 描 红 输 出 。 通 过 关键 字 的 描 红 技 术 ， 可 以 使 查询 结果 更 醒目 ,便于 用 户 进一步 筛选 。 明 日 知 
道 网 站 的 查询 结果 描 红 输 出 的 效果 如 图 SL 


图 9.21 查询 结果 的 关键 字 描 红 输 出 


下 面 将 介绍 PHP 中 搜索 模块 关键 字 描 红 输 出 的 制作 方法 。 在 详细 介绍 描 红 的 制作 步 又 前 ， 首 先 介 
绍 一 下 描 红 的 实现 思路 : 当 用 户 提 交 查 询 关 键 字 到 搜索 页 面 后 ， 通 过 当前 控制 器 对 象 的 _helper 属性 的 
getParam() 方 法 获取 用 户 提 交 的 关键 字 ， 然 后 使 用 PHP 提供 的 explode() 函 数 将 提交 关键 字 以 空格 作为 
分 界 符 保 存 到 数组 中 ， 并 将 该 数组 保存 到 视图 变量 中 ， 最 后 定义 一 个 视图 助手 ， 用 于 将 该 数组 中 所 有 
关键 字 描 红 输 出 。 有 具体 实现 过 程 如 下 : 
(1) 接收 用 户 提交 的 关键 字 ， 并 将 关键 字 保存 到 数组 中 ， 然 后 将 该 数组 赋 给 视图 变量 ， 以 便 在 视 
层 能 够 使 用 该 数组 ， 代 码 如 下 : 


$keywordsStrArray = explode( ', $keywordsStr); // 将 关键 字 字 符 串 用 空格 分 割 保存 到 数组 中 
$this->view->assign('keywordsStrArray', $keywordsStrArray); // 将 数组 赋 给 视图 变量 keywordsStrArray 


(2) 在 views 文件 夹 的 helpers 子 文件 夹 下 建立 SetRed.php 文件 ， 在 该 文件 中 建立 用 于 将 关键 字 
描 红 输出 的 自 定义 助手 Zend_View_Helper_SetRed， 代 码 如 下 : 
倒 程 08 ”代码 位 置 ， 光 盘 \TM\09\known\application\modules\default\views\helpers\SetRed.php 
class Zend_View_Helper_SetRed 


public function setRed ($arrayKeywords, $str) 


$beginFont = ~—'; /关键 字 开始 标识 
$endFont = 一 '; /关键 字 结 束 标识 
foreach [ES as $keyword){ // 遍 历 所 有 关键 字 
$array = array(); 
© preg_match_all(/ . $keyword . i, $str, $array); /查询 与 关键 字 匹 配 的 内 容 ， 并 保存 到 数组 中 
$arrayKey = array_unique($array[0]); // 去 掉 重复 数组 元 素 
foreach ($arrayKey as $key) { // 遍 历 所 有 匹配 关键 字 


$str = str_replace($key, $beginFont . $key . $endFont, $str); // 将 查询 结果 中 匹配 关键 字 的 前 后 
分 别 加 上 关键 字 开 始 标 识 和 结束 标识 


&@ 


} 
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S$str = str_replace($beginFont, '<font color="#FF0000">", $str); /将 开始 标识 替换 成 <font> 标 签 开始 部 分 
$str = str_replace($endFont,'</font>', $str); /将 结束 标识 蔡 换 成 <font> 标 签 结 束 部 分 
return $str; 
} 
} 


< 代码 贴 二 
@ preg_match_all0 函 数 的 语法 格式 为 preg_match_all(string pattem, string subject, array matches, [.int flags]), 该 函数 用 
于 在 subject 中 搜索 与 pattern 给 出 的 正则 表达 式 匹 配 的 内 容 ， 并 将 结果 以 flags 指定 的 顺序 放 到 matches 中 。 

上 述 所 定义 的 视图 助手 包含 关键 字数 组 和 要 描 红 的 字符 串 两 个 参数 。 该 视图 助手 首先 通过 foreach() 
语句 遍历 关键 字数 组 ， 然 后 使 用 preg_match_all() 函 数 提取 要 描 红字 符 串 中 所 有 与 关键 字 相 匹配 的 字符 
串 ， 并 将 匹配 的 字符 串 保 存在 一 个 数组 中 ， 去 掉 重 复元 素 后 ， 将 该 数组 所 有 元 素 的 开始 和 结束 加 上 一 
个 标识 ， 最 后 将 所 有 开始 标识 蔡 换 为 <font color="#FF0000">， 结 束 标识 蔡 换 成 </font>， 最 终 达 到 在 所 
要 描 红 字符 串 的 所 有 与 关键 字 匹 配子 串 的 前 后 加 上 <font> 标 签 , 这 样 可 以 达到 在 输出 该 字符 串 时 , 字符 
串 中 的 匹配 关键 字 描 红 输出 的 效果 。 


Ne 
说 昌 
Zend Framework 框架 中 ， 自 定义 视图 助手 一 般 存放 在 views 目录 下 的 helpers 子 目录 中 , 视 
图 助手 文件 应 以 大 写 开 头 ， 视 图 文件 中 的 类 名 要 以 Zend_View_Helper 作为 前 级 ， 然 后 紧 接 与 视图 文 
件 名 相同 的 字符 作为 完整 类 名 , 视图 文件 中 包含 一 个 与 视图 文件 名 相同 的 方法 , 该 方法 首 字母 要 小 写 。 


(3) 在 search.phtml 文件 中 通过 上 面 介绍 的 视图 助手 将 查询 结果 中 的 关键 字 进 行 描 红 输出 ， 代 码 
如 下 : 


echo $this->setRed($this->keywordsStrArray, htmlspecialchars($this->substr($search[unhtmlcontent], 500))) 


9.6.4 制作 在 线 编辑 器 


以 往 用 HTML 提供 的 文本 域 对 文本 进行 内 容 编辑 ， 一 般 是 通过 JavaScript 在 要 编辑 文本 的 前 后 增 
加 一 些 标 记 ， 然 后 在 输出 这 些 内 容 时 ， 将 这 些 标记 替换 成 相应 的 HTML 标记 〈 如 <font>、<style> 等 ) ， 
这 样 在 显示 内 容 时 ， 就 可 以 出 现 相 应 的 特效 。 但 是 ， 这 种 编辑 器 并 不 能 达到 所 见 即 所 得 的 效果 ， 目 前 
已 经 有 很 多 组 织 为 开发 人 员 提供 了 在 线 文本 编辑 工具 ， 而 且 功能 也 非常 强大 ， 其 中 FckEditor 就 是 较 著 
名 的 在 线 编辑 器 之 一 ， 开 发 人 员 只 需 简单 的 配置 ， 就 可 以 在 项 目 中 应 用 该 编辑 工具 。 不 过 这 些 编辑 器 
由 于 功能 较 强 大 ， 可 能 存在 访问 页 面 时 加 载 速度 慢 、 针 对 性 不 强 等 缺陷 。 

明日 知道 网 站 在 发 表 问 题 和 回复 问题 时 所 使 用 的 在 线 编辑 器 是 通过 JavaScript 编写 的 , 具备 文字 排 
版 、 更 改 文字 样式 、 揪 入 文字 链接 和 插入 图 片 等 功能 ， 完 全 适合 论坛 、 博 客 和 后 台 管 理 中 的 内 容 编辑 
工作 。 图 9.22 为 明日 知道 网 站 所 使 用 的 在 线 编辑 器 。 
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图 9.22 在 线 编辑 器 
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在 讲解 在 线 编辑 器 具体 实现 步骤 前 ， 首 先 讲解 制作 一 个 在 线 编辑 器 的 基本 思路 。 明 日 知道 网 站 所 
使 用 编辑 器 的 编辑 区 域 实质 为 一 个 <iframe> 内 髓 标签 ,默认 情况 下 <iframe> 标 签 是 不 可 编辑 的 ,需要 设 
置 <iframe> 的 document 属性 的 designMode 属性 的 值 为 on， 来 开启 该 标签 的 可 编辑 模式 ， 然 后 使 用 
<iframe> 的 document 属性 的 executeCommand() 方 法 执行 相应 的 命令 标识 即 可 ， 其 中 常用 的 命令 标识 如 


表 9.5 所 示 。 
表 9.5 常用 命令 标识 符 
参数 说 了 明 
BackColor 设置 或 获取 当前 选中 区 的 背景 颜色 
Bold 切换 当前 选中 区 的 粗 体 显示 与 否 
CreateLink 在 当前 选中 区 上 插入 超 链接 ， 或 显示 一 个 对 话 框 允许 用 户 指定 要 为 当前 选中 区 插入 的 超 链接 的 URL 
Cut 将 当前 选中 区 复制 到 剪贴 板 并 删除 
Paste 用 剪贴 板 内 容 覆 盖 当 前 选中 区 
Cop 将 当前 选中 区 复制 到 剪贴 板 
Delete 删除 当前 选中 区 
FontName 设置 或 获取 当前 选中 区 的 字体 
FontSize 设置 或 获取 当前 选中 区 的 字体 大 小 
ForeColor 设置 或 获取 当前 选中 区 的 前 景 〈 文 本 ) 颜色 
Ttalic 切换 当前 选中 区 斜体 显示 与 否 
JustifyCenter _| 将 当前 选中 区 所 在 格式 化 块 置 中 
JustifyLeft 将 当前 选中 区 所 在 格式 化 块 左 对 齐 
JustifyRight | 将 当前 选中 区 所 在 格式 化 块 右 对 齐 
JInsertImage 用 图 像 覆 盖 当 前 选中 区 
Undo 取消 
Redo 重 做 
underline 切换 当前 选中 区 的 下 划 线 显示 与 否 
strikethrough 删除 线 


了 解 在 线 编辑 器 的 基本 思路 后 ， 下 面 将 具体 讲解 在 线 编辑 器 的 制作 步骤 。 

(1) 建立 在 线 编辑 器 UI。 在 线 编辑 器 是 通过 JavaScript 实现 的 ， 所 以 将 HTML 代码 作为 字符 串 ， 
然后 使 用 JavaScript 的 document 对 象 的 write0 方 法 输出 到 浏览 器 即 可 。 由 于 代码 所 占 篇 幅 较 长 ， 这 里 
只 给 出 关键 代码 ， 具 体 如 下 : 

倒 程 09 ”代码 位 置 ， 光盘 \TM\09\known\public_htmlyjs\LzhEditorn\LzhEditorjs 
LzhEditor.prototype.Create = function() { 
this.editorStr += "<div id=\"faceLayer\" style=\"position: absolute; width:445px; z-index: 1; border:1px solid 
#77B7DD; background-color:#F2F9F9; clear:both; display:none;\" onmouseleave=\"this.style.display= 'none\">"; 
…// 省 略 代 码 请 详 见 本 书 附 带 光盘 ， 该 段 代 码 主要 用 于 构建 编辑 器 的 控制 面板 
this.editorStr += "<iframe id=\"editor\" name=\"editon\" width=\"100%\" height=\"" 


+ this.height + "\" scrolling=\"auto\" frameborder=\"0V></iframe>"; // 通 过 iframe 建立 编辑 区 域 


this.editorStr += "<input type=\"hidden\" id=\"" + this.fieldName 


+ name=\" + this.fieldName + "\" value=\"\" />";”// 通 过 隐藏 域 保存 用 户 在 编辑 器 输入 的 内 容 


this.editorStr += "</div>"; 
document.write(this.editorStr); 


@ 
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window.frames["editor"].document.open(); /打开 <iframe> 

window.frames["editor].document.write("<BODY style=\"PADDING-RIGHT: 5px; PADDING-LEFT: Spx; 
FONT-SIZE: 12px; PADDING-BOTTOM: 5px; MARGIN: 0px; PADDING-TOP: 5px\">"+ this.value + "</BODY>"); 
/将 <body> 标 签 写 入 iframe 


window.frames["editor"].document.close(); /关闭 <iframe> 
@ window.frames["editor"].document.designMode = "on"'; /开启 <iframe> 的 编辑 模式 
window.frames["editor"].focus(); // 使 <iframe> 获 得 焦点 
window.frames["editor].document.onkeydown = function(){ ”// 为 <iframe> 添 加 onkeydown 事件 
if (window.frames["editor].event.keyCode == 13) { // 如 果 是 回 车 


window.frames['editor].document.selection.createRange().pasteHTML('<br><!-—->"); 
/<br> 后 必须 有 内 容 才能 换行 ， 所 以 <br> 后 又 加 了 HTML 的 注释 ， 这 样 既 可 以 实现 换行 ， 还 不 会 显示 多 余 内 容 
window.frames['editor].event.returnValue = false; 


} 
} 


Ah 代码 贴 十 
@ 通过 设置 <iframe> 标 签 document 属性 的 designMode 属性 的 值 为 on， 来 开启 <iframe> 标 签 的 编辑 模式 。 


上 述 代 码 中 ， 使 用 <iframe> 标 签 作为 在 线 编辑 器 的 编辑 区 域 ， 默 认 情 况 下 <iframe> 标 签 是 不 可 编辑 
的 ， 所 以 需要 开启 <iframe> 标 签 的 编辑 模式 。 在 开启 <iframe> 标 签 的 编辑 模式 前 ， 需 要 将 <body> 标 签 写 
入 到 <iframe> 内 部 来 设置 编辑 区 域 边框 间距 、 文 字 大 小 等 属性 。 向 <iffame> 中 写 入 <body> 标 签 ， 首 先 需 
要 调用 <ifframe> 标 签 的 document 属性 的 open() 方 法 打开 <iframe>， 然 后 再 通过 该 属性 的 write() 方 法 将 
<body> 标 签 写 入 到 <iframe> 内 部 ， 最 后 使 用 该 属性 的 close() 方 法 关闭 <iframe> 标 签 。 

(2) 建立 完 在 线 编辑 器 的 躯壳 后 ， 下 一 步 就 需要 为 编辑 器 注入 灵魂 。 注 入 灵魂 的 过 程 实质 就 是 让 
在 线 编辑 器 控制 面板 真正 具有 排版 、 设 置 字体 样式 等 功能 ， 即 为 控制 面板 中 各 个 图 标 按钮 的 onClick 
事件 设置 相应 的 方法 。 为 了 提高 代码 重用 率 ， 这 里 定义 一 个 名 为 LzhEditorFormat() 的 方法 ， 该 方法 中 
有 两 个 参数 ， 分 别 为 要 执行 的 命令 标识 〈 即 表 9.5 所 列 出 的 命令 标识 ) 及 执行 该 标识 所 带 的 参数 。 该 方 
法 的 代码 如 下 : 

倒 程 10 ”代码 位 置 ; 光盘 \TM\WO9\known\public_htmIisLzhEditowLzhEditorjs 

function lzhEditorFormat(hc, pa) { 


window.frames["editor"] focus(); /| 编辑 器 获得 焦点 
window.frames["editor].document.selection.createRange(); /创建 编辑 区 域 
if (pa == ™){ 

window.frames["editor"].document.execCommand(hc, false); // 不 需要 指派 参数 
}else{ 


window.frames["editor].document.execCommand(hc, false, pa); // 需 要 指派 参数 
} 


通过 上 述 代 码 可 知 ， 通 过 <iframe> 标 签 的 document 属性 的 execCommand() 方 法 执行 命令 标识 ， 该 
方法 的 语法 格式 如 下 : 


bSuccess = object.execCommand(sCommand [, bUserlnterface] [, vValue]) 


参数 说 明 如 表 9.6 所 示 。 


_ 国 
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表 9.6 execCommand() 方 法 的 参数 说 明 


参 。 数 说 了 明 
sCommand 必要 参数 ， 所 要 执行 的 命令 标识 
bUserInterface | 可 选 参 数 ，boolean 型 ， 默 认为 tlse， 即 不 显示 用 户 接口 
VValue 可 选 参 数 ， 执 行 命令 标识 ， 所 带 参 数 。 例 如 设置 字体 大 小 时 ， 字 号 通过 该 参数 指定 


(3) 通过 上 述 两 步 讲 解 ， 相 信 读 者 已 经 了 解 制作 一 个 在 线 编辑 器 的 基本 思路 ， 最 后 还 需要 讲解 如 
何在 程序 中 调用 该 编辑 器 。 
由 于 该 编辑 器 被 封装 在 一 个 独立 的 js 文件 中 ， 所 以 首先 应 该 通过 <scrip 忆 标签 的 src 属性 包含 该 js 
文件 ， 然 后 在 form 表单 要 显示 编辑 器 的 位 置 使 用 如 下 代码 引入 编辑 器 。 


var lzhEditor = new LzhEditor(name, width, height ,value, baseUrl, uploadFileUrl); 


lzhEditor.Create(); 


// 实 例 编辑 器 


// 创 建 编辑 器 


通过 上 述 代码 就 可 以 将 编辑 器 显示 在 表单 区 域内 ， 但 当 用 户 提交 表单 前 ， 还 需要 通过 如 下 代码 将 
编辑 器 中 的 内 容 赋 给 隐藏 域 ， 表 单 提交 时 不 能 直接 将 编辑 器 中 的 内 容 提 交 给 服务 器 ， 而 是 间接 地 通过 


隐藏 域 实现 。 


lzhEditor.Submit(); 


// 将 编辑 器 中 的 内 容 赋 给 隐藏 域 


以 上 就 是 制作 一 个 在 线 编辑 器 的 全 部 过 程 , 最 后 在 总 结 下 其 开发 思路 : 首先 开启 <iframe> 标 签 的 编 


辑 模 式 ， 然 后 通过 execCommand() 方 法 执行 对 应 的 命令 标识 即 可 对 <iframe> 中 的 文本 进行 编辑 。 


9.7 MVC 框架 结构 搭建 


明日 知道 网 站 采用 Zend Framework 框架 技术 开发 ， 所 以 首先 应 该 设计 并 搭建 出 稳定 、 扩 展 性 强 的 


框架 结构 。 为 了 能 够 掌握 Zend Framework 框架 的 搭建 方法 ， 下 面 通过 一 个 简单 的 实例 对 此 进行 介绍 。 


9.7.1 MVC 开发 模式 介绍 


MVC 分 别 为 英文 单词 model、view 和 
controller 的 首 字母 ， 中 文 含 义 为 “模型 -视图 
-控制 器 ”， 这 种 开发 模式 是 现 阶段 主流 的 项 
目 开 发 模式 之 一 ， 可 使 整个 项 目的 各 个 层次 
独立 出 来 ， 这 样 程 序 开发 人 员 和 美工 人 员 可 
以 对 项 目 进行 同步 开发 ， 而 且 也 为 项 目的 进 
一 步 功 能 扩展 和 日 后 在 生产 环境 下 的 维护 工 
作 提 供 了 极 大 方便 ， 所 以 MVC 开发 模式 也 
是 现 阶 段 各 大 软件 公司 极力 提倡 的 开发 模 
式 ， 是 程序 开发 人 员 的 必修 内 容 。 图 9.23 为 


他 


视 | 
允许 控制 器 选择 视图 
发 送 用 户 输入 给 控制 
器 解释 模型 


一 一 一 > 方法 调用 


模型 
通知 视图 改变 应 用 程 
序 功 能 响应 状态 、 查 
询 封装 应 用 程序 状态 


状态 改变 


+ 一 一 


| 


控制 器 

用 户 动 作 映射 成 模 
型 更 新 、 定 义 应 用 
程序 行为 、 选 择 响 


方法 请 求 
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事件 


图 9.23 MVC 模式 架构 图 
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典型 的 MVC 模式 架构 图 。 
9.7.2 Zend Framework 框架 的 MVC 目录 结构 

与 其 他 主流 PHP 框架 相 比 ，Zend Framewok 是 一 款 相对 灵活 的 MVC 框架 ， 其 搭建 方式 有 多 种 ， 
其 中 明日 知道 网 站 的 MVC 框架 结构 如 图 9.24 所 示 。 从 图 中 可 以 看 出 ， 该 结构 是 一 种 多 模块 结构 ， 即 
将 前 台 和 后 台独 立成 两 个 模块 ， 从 而 更 有 利于 多 人 协作 开发 。 


9.7.3 ”Zend Framework 框架 的 MVC 结构 创建 过 程 


了 解 明日 知道 网 站 的 框架 结构 图 后 ， 下 面具 体 按 步骤 讲解 使 用 Zend Framework 建立 多 模块 MVC 
框架 结构 的 搭建 过 程 。 为 了 便于 学 习 ， 这 里 给 出 一 个 最 基本 的 Zend Framework 框架 的 流程 ， 如 图 9.25 
所 示 ， 之 后 开发 人 员 就 可 以 在 此 基础 上 进行 扩充 。 

项 目 根 目录 
a 
表单 目录 
Pt 
前 台 控制 器 
前 扣 视 图 
各 
视图 脚本 
后 台 和 模块 
后 台 控 制 蝇 
议 图 助手 
视图 肚 本 
a 
强 存 文件 目录 
局 动 文件 
Zend Franerork 共 库 
Be 
人 运行 一 个 最 基本 的 ZF 程序 
项 目 引 导 文 件 
图 9.24 明日 知道 网 站 框架 结构 图 9.25 最 基本 的 Zend Framework 程序 开发 步骤 


(1) 在 public_html 目录 下 建立 URL 重 写 文件 .htaccess， 并 在 该 文件 中 输入 URL 重 写 规则 ， 如 下 
所 示 : 
# 开 启 URL 重 写 
RewriteEngine on 


# 除 扩展 名 为 .js、.css、.gif、.jpg、.png、.bmp 的 文件 外 ， 访 问 其 他 文件 都 转向 到 index.php 引导 文件 
RewriteRule \.(0slcsslgifljpglpnglbmp)$ index.php 


/ 
WE 说 
盟 由 于 .htaccess 文件 没有 文件 主 名 ， 所 以 在 Windows 系统 下 无 法 直接 命名 ， 下 面 介 绍 两 种 用 
户 创建 .htaccess 文件 的 方法 。 
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方法 一 : 通过 Windows 系统 的 copy con 命令 创建 ， 如 图 9.26 所 示 。 创 建 完成 后 使 用 Ctrl+Z 组 合 


键 退出 编辑 模 起 。 


图 9.26 通过 命令 提示 符 创建 .htaccess 文件 
方法 二 : 通过 “记事 本 ”文本 编辑 工具 将 文件 另存 为 后 级 名 为 .htaccess 的 文件 ， 这 里 应 该 注意 保 


存 文件 时 需要 选择 文件 类 型 为 所 有 文件 。 
(2) 在 public_html 目录 下 创建 index.php 引导 文件 ， 代 码 如 下 : 


defined('APPLICATION_PATH') || define(APPLICATION_PATH', realpath(dimame(__FILE_).'/../application’")); 


/应 用 路 径 
defined(APPLICATION_ENV) || define(APPLICATION_ENV., getenv(APPLICATION_ENV) ? getenv (APPLICATION _ 
ENV') : 'project’); /应 用 环境 
$arraylncludePath = array('., realpath(dirname(_FILE__) .人 library)); /指定 工程 包含 目录 
set_include_path(implode(PATH_SEPARATOR, $arrayIncludePath)); /将 指定 路 径 包含 到 工程 中 
require_once 'Zend/Application.php'; /包含 Application.php 文件 


$application = new Zend_Application(APPLICATION_ENV, APPLICATION_PATH . /configs/application.ini); 
/实例 化 Zend_Application 类 
$application->bootstrap()->run(); /调用 启动 文件 并 运行 项 目 


用 路 径 和 应 用 环境 名 ， 这 样 在 wa 然后 将 当前 目录 地 址 “.” 和 Zend 
Framework 类 库 地 址 作为 数组 元 素 保存 在 名 为 SarrayIncludePath 的 数组 中 , 并 使 用 implode(O) 函 数 将 上 述 
路 径 用 PATH_SEPARATOR 常量 连接 ， 最 后 使 用 set_include_path() 方 法 将 路 径 导 入 到 工程 中 ， 从 而 在 


工程 中 就 可 以 直接 找到 这 些 路 径 下 的 文件 。 


Zend Framework 类 库 被 导入 到 工程 后 ， 就 可 以 使 用 require_once 等 文件 包含 语句 包含 Zend 目录 下 


的 Application.php 文件 ， 之 后 实例 化 Zend_Application 类 ， 并 通过 引用 该 类 的 bootstrap() 方 法 调用 启动 


类 ， 


最 终 再 通过 调用 run() 方 法 运行 工程 。 
(3) 在 第 (2) 步 实 例 Zend_Application 类 时 ， 为 类 的 构造 函数 传 入 一 个 名 为 application.ini 的 文 


件 ， 该 文件 即 为 工程 的 配置 文件 ， 本 步 将 介绍 该 文件 的 创建 过 程 ， 其 中 一 个 基本 的 配置 文件 如 下 : 


[project] 
# 设 置 错 误 级 别 
phpSettings.display_startup_errors = 1 
phpSettings.display_errors = 1 
# 设 置 时 区 
phpSetting.date.timezone = 'Asia/Shanghai 
# 配 置 启动 类 
bootstrap.path = APPLICATION_PATH "/Bootstrap.php" 
bootstrap.class = "Bootstrap” 
# 配 置 工程 模块 
resources.frontController.moduleDirectory = APPLICATION_PATH "/modules" 


@ 
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配置 文件 存放 位 置 和 名 称 并 不 固定 ， 只 要 与 实例 化 Zend_Application 类 传递 的 参数 相 一 致 即 可 。 
(4) 在 application.ini 配置 文件 中 指定 启动 类 的 位 置 后 ， 工 程 就 可 以 找到 启动 文件 Bootstrap.php， 
其 关键 代码 如 下 : 


clclass Bootstrap extends Zend_Application_Bootstrap_Bootstrap 


// 基 本 的 启动 类 可 以 不 进行 任何 操作 


从 上 述 代 码 可 知 ,启动 Bootstrap 继承 自 类 Zend_Application_Bootstrap_Bootstrap, 该 类 已 经 实现 最 
基本 的 启动 配置 工作 ， 所 以 在 启动 类 中 不 需 包含 任何 代码 就 可 以 运行 最 基本 的 Zend Framework 应 用 。 
(5) 完成 以 上 步骤 后 ， 就 可 以 创建 默认 控制 器 mdexController， 同 样 一 个 最 基本 的 控制 器 应 该 包 
含 一 个 首页 动作 indexAction， 代 码 如 下 : 


class IndexController extends Zend_Controller_Action 


public function indexAction () /1/ 上 默认 动 作 
{ 

S$this->view->testStr="Hello ZF!"; // 为 视图 变量 赋值 
} 


} 


A 


” 使 用 Zend Framework 时 ， 控 制 器 名 应 该 使 用 Controller 结束 ， 动 作 名 应 以 Action 结束 。 


(6) 完成 默认 控制 器 及 首页 动作 的 创建 后 ， 就 可 以 创建 视图 文件 index.phtml， 为 了 查看 效果 ， 在 
视图 文件 中 仅 输出 首页 动作 中 指定 的 视图 变量 的 值 ， 代 码 如 下 : 


echo $this->testStr; 


(7) 假 设 上 述 工程 所 在 目录 为 Apache 默认 主 目 录 的 test 子 目 录 下 , 就 可 以 通过 如 下 URL 进行 访问 : 

http://localhost/test/public_html 

名 http://localhost/test/public_html/index 

http://localhost/test/public_html/index/index 

正常 情况 下 ，Zend Framework 的 访问 URL 应 该 为 “域名 + 模块 名 + 控制 器 名 + 动作 名 ”， 那 么 为 什 
么 上 述 URL 省 略 了 部 分 内 容 也 可 以 访问 呢 ? 这 是 因为 在 Zend Framework 中 ， 默 认 模块 名 、 默 认 控 制 
器 名 和 首页 动作 名 可 以 省 略 。 

运行 上 述 工程 ， 可 以 在 浏览 器 中 输出 “Hello ZF!”。 


9.8 首页 设计 


网 站 首页 的 内 容 应 该 突出 网 站 主题 及 特色 ， 在 网 站 的 生命 周期 中 起 着 非常 重要 的 作用 ， 本 节 将 以 
明日 知道 网 站 为 例 讲解 如 何 设计 合理 的 、 适 合 网 站 特点 的 首页 。 
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9.8.1 首页 概述 


设计 网 站 首页 要 贴切 网 站 主题 、 突 出 网 站 特色 ， 而 非 复杂 就 可 以 。 考 虑 到 明日 知道 网 站 主要 用 于 
为 编程 相关 工作 人 员 提 供 技术 问题 搜索 和 技术 交流 , 借鉴 互联 网 一 些 知名 的 搜索 工具 , 设计 出 如 图 9.27 
所 示 的 首页 界面 。 明 日 知道 网 站 的 首页 主要 由 网 站 Logo、 语 言 导航 条 、 搜 索 区 、 友 情 链 接 和 版 权 声明 
等 部 分 组 成 ， 页 面 简洁 大 方 、 突 出 核心 ， 便 于 用 户 使 用 。 


图 9.27 明日 知道 网 站 首页 


9.8.2 首页 技术 分 析 


明日 知道 网 站 首页 主要 涉及 两 点 较为 关键 的 技术 , 分 别 为 类 似 Google 搜索 条 的 制作 和 语言 类 别 选 
项 卡 的 制作 ， 其 中 类 似 Google 搜索 条 的 制作 过 程 已 经 在 9.6 节 中 进行 讲解 ， 这 里 不 再 袭 述 ， 下 面 重点 
对 语言 类 别 选项 卡 的 实现 过 程 进行 技术 分 析 。 

与 Visual Basic、C# 等 一 些 可 视 化 编程 语言 相 比 ， 网 页 视图 层 的 一 些 组 件 和 特效 的 制作 需要 通过 编 
码 实现 ， 而 非 简单 的 拖 息 ， 这 样 制 作出 的 组 件 更 具 灵 活性 。 制 作 页 面 特 效 一 般 通 过 JavaScript、Ajax 等 

制作 明日 知道 网 站 语言 类 别 选 项 卡 的 基本 思想 是 : 首先 从 数据 库 中 提取 所 要 显示 语言 类 别 的 名 称 ， 
并 通过 <div> 标 签 显示 在 同一 行 ,然后 为 这 些 <div> 标 签 指定 背景 图 片 ,最 后 定义 如 下 方法 (这 里 用 jQuery 
实现 ) 控制 选项 卡 之 间 的 切换 。 

倒 程 11 代码 位 置 : 光盘 \TM\09\known\application\modules\default\views\scripts\index\index.phtml 


function changeSearchType(x, y) 
{ /该 方法 中 的 两 个 参数 x、y 分 别 为 每 个 选项 卡 按钮 的 ID 和 选项 卡 对 应 语言 的 1D 
$("#st_0").css("background", "url(<?php echo $this->baseUrl(/img/main_dh_btn2.gif)?>)"); /更 改 全 部 选项 
卡 背 景 图 片 
$("#st_0").css("color", "#333333"); // 更 改 全 部 选项 卡 前 景色 
<?php foreach ($this->types as $type):?> 
$("#st_<?php echo $type['tid]?>").css("background", "url(<?php echo S$this->baseUrl(/img/main_dh_ 
btn2.gif)?>)"); // 更 改 语言 类 别 选项 卡 背景 图 片 
$("#st_<?php echo $type[tid]?>").css("color", "#333333"); /更 改 语言 类 别 选项 卡 前 景 
<?php endforeach;?> 
$("#"+x).css("background", "url(<?php echo $this->baseUrl(Vimg/main_dh_btn1.gif)?>)"); /更 改 当 前 选中 
的 选项 卡 背景 图 片 
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$C#"+x).css('"color’, #FFFFFF"); // 更 改 当前 选中 的 选项 卡 的 前 景 
@ S$(#ypeid")val(y); // 更 改 隐藏 域 ypeid 的 值 为 当前 选中 选项 卡 对 应 的 语言 ID 


上 述 代码 中 ， 通 过 jQuery 的 css() 方 法 更 改 <div> 标 签 的 背景 图 片 和 前 景色 ， 该 方法 可 以 传 入 两 个 
参数 ， 第 一 个 参数 为 要 更 改 的 CSS 样式 属性 名 称 ， 如 color、background-color、font-size 等 ， 第 二 个 参 
数 是 为 属性 所 指定 的 值 ， 其 中 第 二 个 参数 可 以 省 略 ， 如 果 设 置 第 二 个 参数 ， 则 为 CSS 样式 重新 赋值 ， 
否则 将 获取 该 CSS 样式 的 值 。 


9.8.3 首页 的 实现 过 程 


明日 知道 网 站 采用 MVC 开发 模式 ， 所 以 其 开发 过 程 也 需要 按 模型 层 、 视 图 层 和 控制 器 层 的 顺序 
进行 。 在 对 首页 进行 技术 分 析 时 已 经 介绍 到 ， 制 作 该 站 首页 需要 从 数据 库 中 提取 所 有 语言 类 别 ， 所 以 
应 该 在 类 别 表 的 模型 Model_DbTable_Bbstype 中 建立 一 个 用 来 查询 全 部 类 别 的 方法 ， 这 里 命名 为 
findAll0。 代 码 如 下 : 

倒 程 12 ”代码 位 置 ， 光 盘 \TM\09\known\application\models\DbTable\Bbstype.php 

public function findAll ($isCache = false) 


$innerSelect1 = $this->getAdapter()->select(); // 所 有 主题 

S$innerSelect1->from('tb_title', 'count(*))->join($this->_name, tb _title.bbstype_id=' . $this->_name . .id', nul)-> 
where($this->_name . "id = tid'); 

$innerSelect2 = $this->getAdapter()->select(); // 今 日 主题 

S$innerSelect2->from('tb_title', 'count(*))->join($this->_name, tb _title.bbstype_id=' . $this->_name . .id', null)-> 
where(date(tb title.addtime)=date(now()))->where($this->_name . ‘id = tid'); 

$innerSelect3 = S$this->getAdapter()->select(); /最 后 主题 ID 

$innerSelect3->from(tb title', 由 tile.id')->join($this->_name, ‘tb_title.bbstype_id=" . $this->_name . ‘id', null)-> 
where($this->_name . ‘id = tid')->order('tb_title.addtime desc')->limit(1, 0); 

S$innerSelect4 = S$this->getAdapter()->select(); /最 后 主题 

$innerSelect4->from(tb title', ‘tb_title.title’)->join($this->_name, ‘tb_title.bbstype_id=" . $this->_name . .id', nul)-> 
where($this->_name . ‘id = tid')->order('tb_title.addtime desc')->limit(1, 0); 

$innerSelect5 = $this->getAdapter()->select(); // 最 后 主题 时 间 

S$innerSelect5->from('tb_title’, 'tb_title.addtime')->join($this->_name, 'tb_title.bbstype_id=" . $this->_name . 
\id', nul)->where($this->_name . ‘id = tid')->order('tb_title.addtime desc’)->limit(1, 0); 

$innerSelect6 = S$this->getAdapter()->select(); /总 回复 

S$innerSelect6->from('tb_reply', 'count(*)')->join($this->_name, 也 _reply.bbstype_id=' . $this->_name . .id', null)-> 
where($this->_name . ‘id = tid'); 

$innserSelect7 = $this->getAdapter()->select(); /最 后 回复 用 户 1D 

S$innserSelect7->from('tb_title', ‘tb_title.user_id')->join($this->_name, tb _title.bbstype_id=' . $this->_name . 
“id, null)->where($this->_name . "id=tid')->order('tb_title.addtime desc’)->limit(1, 0); 


$innserSelect8 = $this->getAdapter()->select(); /最 后 回复 用 户 昵 称 
$innserSelect8->from('tb_user, 'netname')->where('id=(" . $innserSelect7 . )); 
$select = $this->getAdapter()->select(); // 外 层 查询 


$select->from($this->_name, array($this->_name . ‘id as tid' , $this->_name . 'typename' , $this->_name . 
"description' , $this-> name . ‘addtime' , '(' . $innerSelect1 . ') as totaltitle' ，'(' . $innerSelect2 . ') as 
totaltodaytitle' , '(' . $innerSelect3 . ') as lasttitleid' , '(' . $innerSelect4 . ') as lasttitletitle' ,'( . $innerSelect5 . ') as 
lasttitletime' , '( . $innerSelect6 . ') as totalreply' , '( . $innserSelect8 . ') as lasttitlenetname'))->order(" . 
S$this->_name . .addtime asc’); 
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if ($isCache) { // 如 果 缓 存 查询 结果 
if (! $result = $this->_cache->load(strtoupper($this->_name . "findAl))){ 
// 判 断 缓存 文件 是 否 存 在 或 缓存 是 否 过 期 
$result = $this->getAdapter()->fetchAll($select); // 如 果 缓 存 文件 不 存在 或 缓存 过 期 ， 则 重 写 查找 
$this->_cache->save($result, strtoupper($this->_ name .findAll)); 
// 写 入 并 生成 新 缓存 
} 


}else{ 
Sresult = $this->getAdapter()->fetchAll($select); // 不 缓存 则 直接 查找 


上 
return $result; // 返 回 结果 
} 二 二 了 = 


上 述 代 码 中 查询 全 部 语言 类 别 通 过 Zend_Select 并 使 用 子 查询 及 Zend_Cache 缓存 技术 实现 。 


首先 介绍 如 何 使 用 Zend_Select 进行 查询 。 使 用 Zend_Select 进行 查询 ， 需 要 获得 数据 库 适配器 对 
然后 通过 该 对 象 的 select0 方 法 生成 Zend_Select 对 象 ， 代 码 如 下 : 


此 


$select = $this->getAdapter()->select(); 


其 中 , $this 代表 当前 表 的 模型 , 由 于 该 模型 继承 自 Zend_Db_Table 类 , 所 以 通过 $this->getAdapter() 
方法 即 可 获取 数据 库 适 配器 。 获 得 Zend_Select 对 象 后 ， 就 可 以 通过 该 对 象 的 from() 方 法 ， 指 定 查询 的 
表 名 、 字 段 名 ;通过 order() 方 法 指定 记录 排序 方式 。 

那么 如 何 通 过 Zend_Select 进行 子 查 询 呢 ? 其 原理 是 : 首先 定义 一 个 Zend_Select 对 象 作为 子 查 询 ， 
然后 在 外 层 查 询 中 ， 将 该 Zend_Select 对 象 作为 查询 结果 的 一 个 字段 出 现 ， 代 码 如 下 : 

$inSelect = $db->select(); // 内 层 查询 ， 这 里 只 给 出 关键 代码 


$outSelect = $db->select(); /外 层 查询 
$outSelect->from( 表 名 , array( 字 段 1', ' 字 段 2', '(.$inSelect.') as 字段 3' ));// 指 定 外 层 查询 表 名 和 字段 名 


最 后 介绍 如 何 对 查询 结果 进行 缓存 输出 。Zend Framework 框架 中 , 使 用 Zend_Cache 对 数据 进行 组 
存 来 提高 程序 执行 效率 。 在 制作 上 述 查 询 全 部 语言 类 别 方法 时 ， 为 了 降低 对 数据 库 频 繁 查 询 的 次 数 ， 
降低 数据 库 服 务 器 负载 ， 提 交 程序 执行 效率 ， 这 里 使 用 Zend_Cache 对 查询 结果 进行 缓存 。Zend_Cache 
通过 其 静态 方法 factory(0) 获 得 其 实例 ， 代 码 如 下 : 


$cache = Zend_Cache:factory(string frontendName, sting backendName, aray frontendOptions, amay backendOptions); 


参数 说 明 如 表 9.7 所 示 。 
表 9.7 Zend_Cache 的 factory() 方 法 的 参数 说 阴 
属 性 值 说 了 明 
frontendName 必要 参数 ， 缓 存 前 端 名 称 
backendName 必要 参数 ， 缓 存 后 端 名 称 


可 选 参数 ， 缓 存 前 端 选项 
可 选 参数 ， 缓 存 后 端 选 项 


frontendOptions 
backendOptions, 


为 了 更 加 明确 地 了 解 Zend_Cache 的 使 用 方法 ， 下 面 以 本 项 目 为 例 按 步 讲解 Zend Framework 中 组 
存 的 使 用 方法 。 


@ 
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(1) 为 了 便于 以 后 维护 ， 将 缓存 过 期 时 间 和 缓存 文件 路 径 在 application.ini 配置 文件 中 指定 ,代码 
如 下 : 
# 缓 存 时 间 
cache.leftTime = "3600" 


# 缓 存 文件 存储 目录 
cache.cache_dir = APPLICATION_PATH tmp/ 


(2) Zend_Cache 对 象 应 该 被 项 目 全 局 应 用 ， 所 以 Zend_Cache 对 象 在 启动 文件 中 被 初始 化 ， 代 码 


如 下 : 
$config = $this->getOptions();。// 通 过 启动 类 的 getOptions() 方 法 获取 配置 文件 配置 信息 , 并 保存 在 多 维 数组 中 返回 
Zend_Registry::set('config, $config); /| 注册 该 多 维 数组 
$frontOptions = array('leftTime' => $config['cache'][leftTime'] , ‘automatic_serialization' => true); 
/| 缓存 前 端 选项 
$backOptions = array('cache_dir => $config[cache][cache_dir]); /缓存 后 端 选 项 
$cache = Zend_Cache::factory('Core', 'File', $frontOptions, $backOptions); /获取 Zend_Cache 对 象 
Zend_Registry::set('cache', $cache); 1 注册 Zend_Cache 对 象 


(3) 使 用 Zend_Register 的 静态 方法 set0 将 缓存 对 象 注册 后 ， 就 可 以 在 工程 的 其 他 模块 通过 
Zend_Register 的 get() 方 法 获取 缓存 对 象 并 使 用 。 代 码 如 下 : 


private $_cache; // 将 缓存 对 象 声 明 为 私有 成 员 ， 这 样 就 可 以 在 模型 的 其 他 方法 中 使 用 
public function init () 


$this->_cache = Zend_Registry::get(cache'); 。 // 获 取 缓 存 对 象 
> 


(4) 在 类 别 表 模 型 的 findAll0 方 法 中 ， 通 过 缓存 对 象 的 load() 方 法 判断 在 缓存 目录 下 是 否 有 以 该 
方法 参数 为 缓存 ID 的 缓存 文件 并 且 不 过 期 ,如果 存 在 则 从 缓存 文件 中 获取 要 查询 的 内 容 ， 否 则 从 数据 
库 中 查询 内 容 并 保存 在 缓存 文件 中 ， 代 码 如 下 : 

if (! $result = $this->_cache->load(strtoupper($this->_name . "findAII))) {// 判 断 缓存 文件 是 否 存在 或 缓存 是 否 过 期 


$result = $this->getAdapter()->fetchAll($select); // 如 果 缓存 文件 不 存在 或 缓存 过 期 ， 则 重 写 查找 
$this->_cache->save($result, strtoupper($this->_ name .'" findAll)); / 写 入 并 生成 新 缓存 


9.9 用 户 模块 设计 
用 户 作为 网 站 活动 的 参与 者 和 维护 者 ， 是 网 站 能 够 生存 发 展 的 源 动力 。 明 日 知道 网 站 由 于 需要 用 
户 提出 问题 、 解 决 问题 ， 所 以 一 个 合理 的 用 户 模块 显得 十 分 重要 。 
9.9.1 用 户 模块 概述 


明日 知道 网 站 的 用 户 模块 ， 主 要 包括 用 户 注册 、 登 录 、 用 户 信 息 管理 等 功能 。 浏 览 者 如 果 只 进行 
问题 查找 ， 则 无 须 注 册 ， 如 果 没 有 在 该 站 点 查找 到 预期 的 结果 ， 那 么 可 以 将 问题 发 布 到 明日 知道 网 站 


@ 
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中 ， 其 他 用 户 就 可 以 与 之 探讨 、 解 答 ， 这 时 就 需要 用 户 注 册 为 本 站 会 员 。 用 户 访问 明日 知道 网 站 的 流 
程 如 图 9.28 所 示 。 


进入 浏览 


图 9.28 用 户 访问 流程 


9.9.2 用户 模 块 技术 分 析 


为 了 防止 非法 用 户 通 过 恶意 程序 ， 采 用 试探 密码 的 方式 登录 本 站 ， 或 连续 地 向 用 户 表 中 注入 信息 ， 
明日 知道 网 站 的 用 户 登 录 和 注册 表单 中 都 增加 了 验证 码 ， 从 而 可 以 有 效 地 预防 上 述 情况 发 生 。 
在 Web 项 目 中 ， 验 证 码 可 以 有 多 种 形式 ， 如 文字 验证 码 、 图 片 验证 码 、 


语言 验证 码 等 ， 本 站 所 采用 的 是 Web 项 目 中 最 为 常用 的 验证 码 形 式 ， 即 图 ”gy 
4 验证 码 ， 如 图 9.29 所 示 。 E 攻 到 


使 用 PHP 语言 制作 验证 码 图 片 可 以 通过 GD2 函数 绘制 。 考 虑 到 验证 码 
的 高 度 、 宽 度 以 及 图 片 中 文字 的 字体 在 不 同类 型 的 表单 中 可 能 存在 差异 ， 在 
制作 明日 知道 网 站 验证 码 时 ， 单 独 将 验证 码 的 实现 过 程 封 装 成 验证 码 类 。 那 么 ， 验 证 码 类 是 如 何 实现 
的 呢 ? 下 面 对 此 进行 讲解 。 

(1) 命 名 该 类 为 Plugin_Util ValidateCode, 这 样 命名 的 原因 是 如 果 将 该 验证 码 类 保存 在 application 

目录 下 的 util 子 目录 中 , 并 将 该 文件 命名 为 ValidateCode.php, 那么 在 需要 使 用 该 验证 码 类 时 ,根据 Zend 
Framework 的 设计 特点 ， 不 需要 使 用 include 等 包含 语句 ， 就 可 以 直接 使 用 该 验证 码 类 。 

定义 该 验证 码 类 的 数据 成 员 ， 代 码 如 下 : 

倒 程 13 ”代码 位 置 : 光盘 \TM\09\known\public_html\known\application\plugins\Util\ValidateCode.php 


图 9.29 ”图片 验证 码 


private $_width; /宽度 

private $_height; // 高 度 

private $_codeStr; // 验 证 码 

private $_fontType; /| 设置 字体 类 型 ，0 表示 粗 体 ，1 表示 简体 
private $_img; /图 像 句柄 


(2) 定义 验证 码 类 的 构造 方法 ， 实 现 验证 码 类 的 初始 化 ， 代 码 如 下 : 
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倒 程 14 ”代码 位 置 : 光盘 \TM\09\known\public_html\known\application\plugins\Util\ValidateCode.php 
public function __construct ($width, $height, $codeStr = "0000", $fontType = 0) 


{ 
S$this->_width = $width; // 验 证 码 宽度 
S$this->_height = $height; // 验 证 码 高 度 
S$this->_codeStr = substr($codeStr, 0, 4); // 验 证 码 的 值 
S$this->_fontType = $fontType; /字体 


(3) 定义 类 的 私有 方法 _getColor0)， 用 于 生成 颜色 句柄 ， 代 码 如 下 : 
全 性 15 ”代码 位 置 ， 光盘 \TM\09\known\public_html\known\application\plugins\Util\ValidateCode.php 
private function _getColor ($x, $y) 


{ 

S$r = mt_rand($x, $y); / 红 

$g = mt_rand($x, $y); / 绿 

$b = mt_rand($x, $y); / 蓝 

return imagecolorallocate( $this->_img, $r, $g, $b); /返回 颜色 句柄 
} 


(4) 定义 私有 方法 _init0)， 用 于 创建 图 片 验 证 码 对 象 ， 代 码 如 下 : 
倒 程 16 ”代码 位 置 ， 光 盘 \TM\09\known\public_html\known\application\plugins\Util\ValidateCode.php 


private function _init () 


$this-> img = imagecreate($this->_width, $this->_height); // 创 建 图 像 
} 
(5) 在 创建 验证 码 图 片 对 象 的 基础 上 创建 验证 码 体 ， 包 括 设置 验证 码 的 填充 色 、 边 框 、 验 证 码 上 
的 文字 内 容 及 字体 等 ， 其 实现 代码 如 下 : 
倒 程 17 代码 位 置 ， 光 盘 \TM\09\known\public_html\known\application\plugins\Util\ValidateCode.php 


private function _build () 


imagefill($this->_img, 0, 0, $this->_getColor(150, 250)); /为 图 像 填充 背景 色 
imagerectangle($this->_ img, 0, 0, $this->_width - 1, $this->_height - 1, $this->_getColor(50, 150)); 
/创建 一 个 矩形 作为 验证 码 的 边框 


if ($this->_fontType == 0){ /设置 验证 码 文字 的 字体 
S$fontFileName = 'ARIALBI.TTF'; 
}else{ 
S$fontFileName ='ARIALN.TTF'; 
人 
for ($i = 0; $i < strlen($this->_codeStr); $i ++){ /绘制 文字 
© imagettftext($this->_img, mt_rand(12, 24), 0, ($this->_width) / 4 * $i, mt_rand(20, $this->_height - 5), 
S$this->_getColor(10, 180), APPLICATION_PATH . /resources/ . $fontFileName, substr($this->_codeStr, $i, 1)); 
} 
for ($i= 0; $i < 15; $i ++) { /| 绘制 15 条 干扰 线 
@ imageline($this->_img, mt_rand(0, $this->_ width), mt_rand(0, $this->_height), mt_rand(0, $this->_width), 


mt_rand(0, $this->_height), $this->_getColor(110, 210)); 
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a 


< 代码 贴 二 
@ imagettftext() 方 法 : 用 于 按 指 定 的 字体 输出 字符 囊 。 
四 imageline() 方 法 : 用 于 绘制 直线 。 
(6) 创建 show0 方 法 用 于 显示 图 片 ， 代 码 如 下 : 
全 性 18 ”代码 位 置 ， 光盘 \TM\09\known\public_html\known\application\plugins\Util\ValidateCode.php 
public function show () 


{ 
header('content-type:image/png'); ll 设置 输出 图 片 格式 


Sthis->_init(); // 图 片 初始 化 
S$this->_build(); 3 
imagepng($this->_ img); 


9.9.3 用 户 注 册 的 实现 过 程 


通过 用 户 注册 ， 可 将 用 户 的 基本 信息 记录 下 来 ， 便 于 用 户 参 与 站 
内 活动 及 用 户 之 间 的 交流 。 明 日 知道 网 站 的 注册 表单 如 图 9.30 所 示 ， 
该 注册 表单 主要 包括 用 户 的 昵称 、 登 录 密 码 、E-mail 地 址 、 联 系 电话 
和 所 在 地 等 信息 。 

了 解 该 注册 表单 的 结构 后 ， 下 面 讲解 该 表单 的 实现 过 程 。 

(1) 建立 用 户 注册 表单 。 明 日 知道 网 站 的 注册 表单 是 通过 Zend 
Framework 的 表单 视图 助手 实现 的 ， 并 通过 jQuery 技术 对 表单 数据 进 
行 校 验 。 注 册 表 单 UI 的 实现 代码 请 参见 本 书 附带 光盘 。 

(2) 保存 用 户 注册 信息 ， 并 向 注册 用 户 的 邮箱 中 发 送 注册 成 功 提 
示 。 当 用 户 填写 完 注 册 信 和 


EE 村 


\， 并 提交 注册 表单 后 ， 用 户 注册 信息 将 被 图 9.30 用 户 注册 表单 
提交 到 用 户 控制 器 的 注册 动作 中 进行 处 理 ， 代 码 如 下 : 


倒 程 19 ”代码 位 置 ， 光 盘 \TM\09known\applicationmodules\default\controllers\UserController.php 


if ($this->_request->isPost()) { 1/ 如果 用 户 通过 POST 方法 提交 表单 
$request = S$this->_request; /获取 当前 控制 器 对 象 的 POST 方法 
$arrayProvice = Plugin_Util_ProvinceAndCityFactory::getProvince(); // 获 取 用 户 所 在 省 份 
$p = $arrayProvice[$request->getParam('p')]; 
$arrayCity=Plugin_Util_ProvinceAndCityFactory::getCitypByProvinceArrayIndex($request->getParam('p')); 
$c = $arrayCity[$request->getParam('c')]; /获取 用 户 所 在 市 


$config = Zend_Registry::get('config'); /获取 Zend_Config 对 象 

$arraySmtpConfig = array(auth' => ‘login' ，userame' => $config['mail][usemame'] ，'password' => 
$config[mail][password']); /邮件 配 置信 息 

S$transport = new Zend_Mail_ Transport_Smtp($config[ mail][host], $arraySmtpConfig);// 构 建 邮件 传输 对 象 
@ $mail=newZend Mail(utf-8"); // 构 建 Zend_Mail 对 象 ， 并 设置 字符 编码 


使 
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$mail->setSubject($config[mail][subject]); /邮件 主题 
$mail->setBodyHtml(file_get_contents($config['mail][bodyPath"])); /HTML 邮件 内 容 
$mail->setFrom($config['mail"][from'], $config[mailrname]); /发 件 人 
$mail->addTo(trim($request->getParam('email)));  // 收 件 人 
ty{ 
$mail->send($transport); /发 送 邮 件 
} catch (Zend_Exception $e){ 
$e->getMessage(); 
} 
/保存 用 户 信息 到 数据 库 
$arrayUserlnfo = array(netname' => trim($request->getParam(netname'")) , ‘password' => md5(trim($request-> 
getParam('‘password'))) , ‘email' => trim($request->getParam(email)) , ‘tel' => trim($request->getParam('tel)) ， 
"pc' => $p . ~. $c , regtime' => date(Y-m-d H:i:s') , face' => " , ‘pubtimes' => 0 ,replytimes' => 0 , 'score' => 20 ， 
"usertype' => 0); /用 户 注册 信息 所 构建 的 数组 
try{ 
$this->_userModel->insert($arrayUserlnfo); /保存 用 户 注册 信息 
} catch (Zend_Exception $e){ 


$e->getMessage(); 
} 
S$this->_sessionNamespace->netname = trim($request->getParam('netname’)); 
S$this->_redirect('/user/register-success'); /定向 到 用 户 注 册 成 功 提示 页 面 
exit(); 
} 
Ah 代码 贴 十 


@ 为 Zend_Mail 的 构造 方法 指定 utf-8 这 个 参数 的 作用 是 ， 设 定 邮 件 编码 为 utf-8 编码 。 
上 述 代 码 保存 用 户 注册 信息 ， 主 要 通过 用 户 表 模型 Model_DbTable_User 的 insert0 方 法 实现 , 该 方 
法 的 语法 格式 如 下 : 


insert(array insertArray) 


该 方法 中 主要 包含 一 个 关联 数组 参数 ， 该 数组 的 键 名 为 用 户 表 的 字段 名 ， 键 值 为 用 户 输入 的 注册 
信息 。 

为 了 提高 网 站 形象 ， 提 高 用 户 回访 数量 ， 在 保存 用 户 注册 信息 前 ， 向 注册 用 户 所 填写 的 邮箱 中 发 
送 一 封 注册 成 功 提示 邮件 。 发 送 邮 件 功能 是 通过 Zend_Mail 实现 的 ， 使 用 Zend_Mail 发 送 邮 件 相对 简 
单 ， 实 例 化 Zend_Mail 对 象 后 ， 只 需 通过 调用 该 类 的 几 个 方法 ， 为 邮件 指定 主题 、 内 容 、 发 件 人 和 收 


件 人 等 信息 ， 最 后 再 通过 send() 方 法 将 邮件 发 送出 去 。Zend_Mail 类 中 常用 的 方法 说 明 如 表 9.8 所 示 。 
表 9.8 Zend_Mail 类 中 常用 的 方法 说 明 
属 性 什 说 了 明 
setSubject() | 设置 邮件 主题 
setBody() | 设置 文本 格式 的 邮件 内 容 
setBodyHtml 设置 HTML 格式 的 邮件 内 容 
setFrom() 设置 发 件 人 地 址 
addTo0) 设置 收 件 人 地 址 
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9.9.4 用 户 登 录 的 实现 过 程 


用 户 注册 成 功 后 ， 以 后 访问 本 站 就 可 以 直接 以 已 注册 的 账号 登录 。 明 日 知道 网 站 的 用 户 登 录 页 面 
如 图 9.31 所 示 。 


9.31 用 户 登 录 页 面 
下 面 建立 用 户 登录 表单 。 明 日 知道 网 站 的 登录 表单 使 用 Zend_Form 制作 ， 代 码 如 下 : 
倒 程 20 ”代码 位 置 ， 光 盘 \TM\09\known\application\forms\User\login.php 


class Form_User_Login extends Zend_Form // 定 义 用 户 登录 表单 ， 继 承 自 Zend_Form 


public function __construct ($options = null) 


parent::_construct($options); // 调 用 父 类 构造 函数 
$this->setName(form_login') /表单 名 称 
->setMethod('post') /表单 提交 方法 
->setAction($options['baseUrl] . "user/login') ”// 表 单 提 交 地 址 
->addAttribs(array( 
'style' => 'margin:0px; padding:0px' /表单 样式 
); 
[9 S$this->addElements(array( /为 表单 增加 用 户 昵 称 录入 文本 框 
new Zend_Form_Element_Text(netname', array( 
‘required' => true, /指定 该 表单 元 素 是 必要 的 
"abel' => ' 了 昵称: ， // 元 素 标签 
'attribs' => array( // 元 素 属性 


"class' => 'input_login_form', 
‘style' => 'position:absolute; le 化 32px; top:0px; width:180px; height:18px; line-height:18px' 


eh => array('StringTrim'), // 过 滤 掉 首尾 空格 
walidators' => array( 
array('NotEmpty', true, array('messages' =>'。 请 输入 登录 昵称 ')) // 用 户 昵 称 非 空 验证 
4 corators' => array( /元素 装饰 器 
ViewHelper， 
array(HtmiTag', array('tag'=>'dd')), 
array(Label', array(tag' =>'dt)) 


) 
))， 
new Zend_Form_Element_Password(password', array(  // 为 表单 增加 用 户 密码 录入 文本 框 
‘required' => true, // 指 定 该 表单 元 素 是 必要 的 
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"abel' => ' 密 码 :，'， // 元 素 标签 


'attribs' => array( /密码 框 属性 
class' => 'input_login_form', 


)， 


'filters' => array('StringTrim'), // 过 滤 掉 首尾 空格 
‘validators' => array( // 登 录 密码 非 空 验证 
array(NotEmpty', true, array(messages' => '。 请 输入 登录 密码 ')) 

)， 
"decorators' => array( /| 元素 装饰 器 
ViewHelper， 


array(HtmlTag', array(tag' => 'dd ))， 
array(Label', array(tag' => "dt)) 
) 
))， 


new Plugin_Form_Element_Vcode(vcode' array( /验证 码 
'required' => true, /指定 该 表单 元 素 是 必要 的 
"abel => ' 验 证 码 : ， // 元 素 标签 
'attribs' => array( // 验 证 码 属性 


‘textClass' => 'input_login_form', 


style' => 'position:absolute; le 化 32px; top:0px; width:180px; height:18px; line-height: 18px;" 


‘textStyle' =>'position:absolute; left32px; top:0px; width:60px; height: 18px; ineheight18px; ', 


'imageStyle' =>'position:absolute; left:100px; top:Opx;", 
'spanStyle' => 'position:absolute; left:42px; top:37px; height:18px', 
'aClass' => 'a4', 
functionName' => 'changeValidateCode() 
)， 
'filters' => array('StringTrim'), // 过 滤 掉 首尾 空格 
walidators' => array( 


array(NotEmpty', true, array('messages' => '。 请 输入 验证 码 '))， /验证 码 非 空 验证 


new Plugin_Validate_VcodeRight() /验证 码 正确 性 验证 
), 
'decorators' => array( // 元 素 装 饰 器 
ViewHelper， 


array(HtmilTag', array('tag' => 'dd', 'style' => 'height:55px;')), 
array(Label', array('tag' => 'dt)) 
} 
))， 
new Zend_Form_Element_Image(submitlImage', array( /提交 图 片 按钮 


"required' => false, // 该 元 素 值 可 为 空 
"label' => "， 
'src' => $options[baseUrl] . /img/btn_login.gif', /图片 地 址 


"attribs' => array( 
"style' => 'position:absolute; left:30px; top:0px;" 
) 
)) 


< 代码 贴 十 
@ addElements() 方 法 : 用 于 将 表单 元 素 对 象 添加 到 表单 中 。 


使 用 Zend_Form 制作 表单 ， 除 可 构建 表单 UI 界面 外 ， 还 可 以 同时 实现 用 户 输入 内 容 的 过 滤 、 验 


多 
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证 等 操作 ， 从 而 可 以 大 幅度 地 提高 程序 开发 效率 。 使 用 Zend_Form 构建 表单 组 件 的 构造 方法 一 般 有 两 
个 参数 ， 分 别 为 表单 元 素 名 称 和 指定 该 元 素 属性 的 关联 数组 ， 该 数组 常用 键 值 说明 如 表 9.9 所 示 。 


表 9.9 Zend_Form 表单 元 素 常用 键 值 说 明 


属 性 值 说 了 明 

required 用 于 指定 表单 元 素 值 是 否 是 必要 的 
label 表单 元 素 标签 

attribs 表单 元 素 属性 

filters 对 表单 元 素 进行 过 滤 

validators 对 表单 元 素 进 行 校 验 

decorators 对 表单 元 素 UI 进行 装饰 


9.10 发 表 问 题 模块 设计 


当 用 户 在 明日 知道 网 站 搜索 问题 时 ， 如 果 没 有 查找 到 预期 结果 ， 可 以 将 要 解决 的 问题 发 表 到 该 
其 他 用 户 就 可 以 对 问题 进行 解答 或 探讨 。 


9.10.1 发 表 问题 模块 概述 


如 果 用 户 已 经 注册 为 本 站 会 员 ， 并 成 功 登 录 本 站 ， 就 可 以 通过 明日 知道 网 站 的 发 表 问题 模块 将 问 
题 发 表 出 去 ， 其 他 用 户 就 可 以 回答 该 问题 ， 最 终 可 以 帮助 提问 用 户 获得 问题 的 最 佳 答案 。 明 日 知道 网 
站 的 发 表 问题 表单 主要 包括 问题 主题 、 问 题 类别 、 问 题 一 

的 内 容 描述 、 附 件 、 验 证 码 等 表单 元 素 , 如 图 9.32 所 示 。 


9.10.2 ”发 表 问 题 模块 技术 分 析 


发 表 问 题 模块 中 ， 应 该 掌握 的 主要 技术 包括 在 线 
编辑 器 的 制作 、 图 片 验证 码 的 制作 、 文 件 上 传 的 实现 
过 程 。 其 中 ， 在 线 编辑 器 的 制作 及 图 片 验证 码 的 制作 
过 程 已 经 在 前 面 章节 中 进行 重点 讲解 ， 本 节 介绍 通过 
Zend Framework 实现 文件 上 传 的 过 程 。 


ET TIE TYE ET EE ELITE | 


(1) 用 户 发 表 问 题 表 单 是 通过 Zend_Form 实现 
的 ， 所 以 需要 使 用 Zend_Form_Element_File 类 建立 文 
2 = .32 发 表 问题 表 
件 上 传 域 ， 代 码 如 下 : RY 
倒 程 21 ”代码 位 置 ， 光盘 \TM\09\known\application\forms\Bbs\Pubtitle.php 
new Zend_Form_Element_File(file1', array( /文件 选择 域 
required' => false, /指定 文件 上 传 域 并 非 必要 
"abel' => ' 附 件 : ,， /标签 
'maxFileSize' => '30000000', /上 传 文件 大 小 
"description' => ' 请 选择 要 上 传 的 附件 '， /| 描述 
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"attribs' => array( /文件 上 传 域 属性 
'size' => "60 
"class' => 'input_pubtitle form' 

)， 

"validators' => array( // 非 空 验证 
array(NotEmpty', false, array('messages' => ' 请 输入 内 容 ')) 

), 

"decorators' => array( /装饰 器 
'File’, 


array(Description', array(tag' => 'dt')), 
array('HtmlTag', array(tag' => 'dd') ), 
array(Label', array(tag' => "dt)) 
) 
), 


在 讲解 用 户 登录 表单 时 , 已 经 详细 介绍 Zend_Form 表单 及 表单 元 素 常 用 属性 的 含义 ， 上 述 代码 中 ， 
使 用 “'maxFileSize' => '30000000'” 指 定 所 允许 上 传 附件 的 最 大 值 ， 运 行 发 表 问 题 模块 ， 上 述 代码 将 转 
换 为 如 下 HTML 代码 : 


<input type="hidden" name="MAX_FILE_SIZE" value="30000000" id="MAX_FILE_SIZE"> 
<input type="file" name="file1" id="file1" size="60" class="input_pubtitle_form"> 


(2) 当 用 户 提交 发 表 问 题 表 单 后 ， 用 户 提交 的 内 容 将 被 提交 到 QuestionController 控制 器 的 
pubtitleAction 动作 进行 处 理 ， 其 中 用 户 实现 文件 上 传 的 关键 代码 如 下 : 
倒 竹 22 代码 位 置 ， 光盘 \TM\09\known\application\modules\default\controllers\QuestionController.php 


$adapter = new Zend_File_Transfer_Adapter_Http(); /构建 上 传 适配器 


$upfileName ="; /上 传 后 ， 保 存在 服务 器 中 的 文件 名 
if ($adapter->getFileName(file1')!= null) { /如果 用 户 已 经 选择 了 上 传 文件 
$upfileDir = $this->_config[bbs]rupfilesdir]; // 从 配置 文件 中 读 取 上 传 文件 保存 目录 
if (! is_dir($upfileDin)) { /| 判断 上 传 目录 是 否 存 在 ， 如 果 不 存 在 则 创建 该 目录 
mkdir($upfileDir); 
} 
$adapter->setDestination($upfileDir); // 为 上 传 适配器 设置 上 传 目录 


$arrayOldFileName = array_reverse(explode(… basename($adapter->getFileName('file1")))); 
// 提 取 上 传 文件 原来 的 名 字 
$extendsFileName = … . $arrayOldFileName[0]; /获取 上 传 文件 扩展 名 
$upfileName = date(YmdHis') . mt_rand(1000, 9999) . $extendsFileName; /为 上 传 文件 指定 新 名 称 ， 用 时 
将 和 一 个 4 位 随机 数组 成 
$upfilePathAndName = $upfileDir . / . SupfileName; /文件 在 服务 器 中 保存 目录 及 名 称 
$adapter>addFilter(Rename', array(target => $upfilePathAndName , 'overwrite' => true), file1");// 更 改 上 传 
文件 名 
$adapter>addValidator('Size', false, array('min' => '0kB' , ,max' => '2MB' , 'bytestring' => false , ,messages' 


=> '。 您 所 上 传 的 文件 不 能 超过 2MB')); /指定 上 传 文件 的 范围 
$adapter->addValidator('ExcludeExtension', false, array(php' ,',exe', 'messages' => '。 您 上 传 的 文件 类 型 不 

允许 ')); /指定 上 传 文件 扩展 名 

} 


上 述 代 码 实现 文件 上 传 是 通过 Zend_File 实现 的 ， 由 于 本 模块 上 传 方式 采用 HTTP 方式 , 所 以 实现 
文件 上 传 主要 通过 Zend_File_Transfer_Adapter_Http 这 个 适配器 实现 , 该 适配器 常用 的 方法 说 明 如 表 9.10 
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所 示 。 


表 9.10 Zend_File_Transfer_Adapter_Http 适配器 常用 的 方法 说 明 


属 性 值 说 了 明 
getFileName() 获取 上 传 文件 的 名 称 
setDestination() 指定 上 传 文件 保存 到 服务 器 中 的 目录 
addFilter 为 上 传 增加 过 滤 规 则 
addValidator() 为 上 传 增加 校 验 规则 


9.10.3 发表 问题 模块 实现 过 程 


制作 明日 知道 网 站 发 表 问 题 模块 ， 主 要 包括 发 表 问题 表单 视图 的 建立 和 保存 问题 内 容 两 个 关键 


环节 。 


(1) 明日 知道 网 站 发 表 问 题 表单 使 用 Zend_Form 实现 ， 通 过 Zend_Form 实现 表单 的 制作 可 以 简 
化 对 提交 数据 的 过 滤 、 校 验 、UI 布局 的 操作 ， 从 而 提高 开发 效率 ， 便 于 开发 人 员 分 层 开 发 及 日 后 生产 


环境 下 的 维护 工作 。 本 站 实现 发 表 问 题 表单 的 代码 如 下 : 


倒 答 23 ”代码 位 置 ， 光盘 \TM\09\known\application\forms\Bbs\Pubtitle.php 


class Form_Bbs_Pubtitle extends Zend_Form 


public function __construct ($options = null) 


parent::_construct(null); // 调 用 父 类 构造 方法 
$this->setName(form_pubtitle') /表单 名 称 
->setMethod('post’) /表单 提交 方法 
->setAction($options[baseUrl]./question/pubtitle') ”// 表 单 提交 地 址 
->addAttribs(array( // 表 单 属性 
"enctype' => 'multipart/form-data’, 
'style' => 'margin:0px; padding:0px;', 
)); 
S$this->addElements(array( 
new Zend_Form_Element_Text(‘title', array( 
1/ 省略 代码 用 于 设置 问题 主题 元 素 的 属性 


))， 
new Zend_Form_Element_Select(bbstypeid', array( 


))， 


new Plugin_Form_Element_Editor( content', array( 


))， 
new Zend_Form_Element_File(file1, array( 


))， 


new Plugin_Form_Element_Vcode(vcode', array( 


))， 


// 省 略 代码 用 于 设置 问题 类 别 下 拉 列 表 属 性 
// 省 略 代码 用 于 设置 问题 内 容 编辑 器 的 属性 
/文件 选择 域 

// 省 略 代码 用 于 设置 文件 选择 框 的 属性 


// 省 略 代码 用 于 设置 验证 码 的 属性 


new Zend_Form_Element_Button(submitButton', array( 
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// 省 略 代 码 用 于 设置 提交 按钮 的 属性 


), 

new Zend_Form_Element_Hidden('titleid', array( // 在 编辑 问题 时 ， 该 隐藏 域 用 于 保存 问题 ID 的 值 
'value' => $options['titleid] 

)) 


} 


上 述 Form_Bbs_Pubtitle 类 继承 自 Zend_Form 类 ， 这 样 该 类 就 可 以 使 用 Zend_Form 类 的 所 有 方法 ， 
如 使 用 setName() 方 法 设置 表单 名 称 、setMethod() 方 法 设置 表单 提交 方法 、setAction() 方 法 设置 表单 提 
交 地 址 等 ， 其 中 Zend_Form 表单 元 素 常 用 类 说 明 如 表 9.11 所 示 。 
表 9.11 Zend_Form 表单 元 素 常用 类 说 明 
属 性 值 说 了 明 


Zend Form Element Text 生成 文本 框 

Zend Form Element Select 生成 下 拉 列 表 框 
Zend_Form Element File 生成 文件 选择 域 
Zend Form Element_Button 生成 按钮 

Zend Form Element Hidden 生成 隐藏 域 

Zend_ Form Element Checkbox 生成 复 选 框 

Zend Form Element Image 生成 图 片 按钮 
Zend Form Element Password 生成 密码 框 

Zend_ Form Element Radio 生成 单 选 按钮 
Zend Form Element Reset 生成 “ 重 置 ”按钮 
Zend Form_Element_Submit 生成 “提交 ”按钮 
Zend Form Element Textarea 生成 文本 域 

Zend Form_ Element Text 生成 文本 框 


(2) 用 户 提 交 表 单 后 ， 输 入 的 内 容 将 被 提交 到 QuestionController 控制 器 的 pubtitleAction 动作 并 


保存 。 保 存 用 户 提交 内 容 的 核心 代码 如 下 : 


倒 程 24 ”代码 位 置 ， 光盘 \TM\09\known\application\modules\default\controllers\QuestionController.php 


$pubUser = $this->_userModel->findByNetname($this->_auth->getldentity()); // 获 取 发 帖 用 户 信息 
$arraylnsert = array('title' => $formDataftitle] , 'content => $formData['content] , ‘unhtmlcontent => $this->view-> 


unHtml($formData[ content]) , 


'addtime' => date(Y-m-d H:i:s') , 'istop' => false , user_id' => $pubUserfid] , 'bbstype_id' => $formData [bbstypeid] , 


"browse' => 0 , 'filename' => 
$upfileName , 'isjh' => false); 


S$this->_titleModel->insert($arrayInsert); 
S$lastTitle = $this->_ titleModel->findLastByUserid($pubUser['id]); 


的 信息 


S$this->_redirect('question/thread/param/" . $lastTitle["id"]); 


// 使 用 问题 模型 的 insert() 方 法 ， 将 问题 信息 保存 到 数据 库 
// 查 找 该 用 户 最 后 发 帖 信息 ， 即 刚 插入 帖子 


// 使 页 面 定位 到 该 帖 的 详细 信息 页 


上 述 代 码 首先 通过 用 户 模型 的 findByNetname() 方 法 获得 当前 登录 用 户 的 信息 ， 然 后 将 问题 信息 保 
存 到 数据 库 的 tb_title 表 中 ， 成 功 保存 问题 信息 后 ， 将 页 面 定 向 到 所 发 问题 的 详细 信息 页 。 
® 
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9.11 回复 问题 模块 设计 


用 户 成 功 注册 为 明日 知道 网 站 会 员 后 ,不 仅 可 以 发 表 问题 还 可 以 回复 问题 ， 与 其 他 用 户 进行 讨论 。 
9.11.1 回复 问题 模块 概述 


明日 知道 网 站 的 问题 回复 表单 采用 Zend_Form 制作 ， 主 要 包括 回复 主题 、 回 复 内 容 和 验证 码 等 表 
单元 素 ， 如 图 9.33 所 示 。 
ET 
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图 933 问题 回复 表单 


9.11.2 回复 问题 模块 技术 分 析 


回复 问题 模块 中 ， 应 该 掌握 的 主要 技术 包括 在 线 编辑 器 的 制作 和 使 用 方法 、 图 片 验证 码 的 制作 方 
法 ， 以 及 问题 引用 的 实现 ， 其 中 在 线 编辑 器 和 图 片 验 证 的 实现 方法 在 上 面 已 经 介绍 ， 下 面 介绍 明日 知 
道 网 站 问题 引用 的 实现 过 程 ， 关 键 代 码 如 下 : 

$value = '<FIELDSET><LEGEND> 引 自 ; 楼 主 </LEGEND> 引 用 的 内 容 </FIELDSET>'; 


将 上 述 代码 作为 编辑 器 的 value 值 ， 将 在 编辑 器 显示 如 图 9.34 所 示 的 效果 。 通 过 上 述 代码 可 知 ， 
引用 可 以 通过 HTML 标签 的 <FIELDSET> 标 签 实现 ，<FILELDSET> 标 签 用 于 在 文本 或 其 他 元 素 外 绘制 
一 个 边框 ， 如 果 在 该 标签 内 使 用 <LEGEND> 子 标签 指定 标题 ， 则 在 边框 左上 侧 还 会 显示 指定 的 标题 。 


引 自 : 楼 主 | 


| 我 是 今年 刚 工作 的 程序 员 ， 对 一 些 代码 的 运用 还 不 是 很 熟悉 ， 对 代码 的 书写 还 是 不 很 规范 。 公 司 要 我 开 
| 发 一 个 项 目 ， 我 有 点 模 不 看 藉 绪 ， 朋 友 买 的 编程 词典 我 看 了 一 下 ， 就 得 很 实用 ， 旺 面 的 代 玛 是 编辑 好 
的 ， 直 接 就 可 以 实用 ， 省 了 我 不 少 的 时 间 啊 ， 经 理 还 说 我 做 的 快 ， 给 我 发 了 奖金 ， 有 机 会 一 定 要 请 你 们 
吃饭 呢 


图 9.34 问题 引用 


@ 
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9.11.3 回复 问题 模块 实现 过 程 


问题 回复 模块 ， 主 要 包括 回复 表单 的 设计 和 保存 回复 信息 两 个 主要 过 程 。 
(1) 明日 知道 网 站 的 问题 回复 表单 采用 Zend_Form 制作 ， 关 键 代码 如 下 : 
全 性 25 ”代码 位 置 ， 光盘 \TM\09\known\application\modules\default\controllers\QuestionController.php 


class Form_Bbs_Reply extends Zend_Form 


public function __construct ($options = null) 
{ 


parent::_construct(null); ll 调用 父 类 构造 函数 
$this->setName(form_reply) /设置 回复 表单 名 称 
->setMethod('post') /设置 表单 提交 方法 
->setAction($options[baseUr'].'/question/thread/param/" . $options['titleid] . -1-T) 
// 设 置 表单 提交 地 址 
->addAttribs(array( /设置 提交 表单 属性 
'style' => margin:0px; padding:0px;', 
)); 
S$this->addElements(array( 
new Zend_Form_Element_Text('title', array( 


i /省略 代 码 用 于 设置 回复 主题 文本 框 属性 
))， 
new Plugin_Form_Element_Editor( content', array( 
i /| 省略 代码 用 于 设置 编辑 器 属性 
))， 
new Plugin_Form_Element_Vcode(vcode', array( 
/| 省略 代码 用 于 设置 验证 码 属性 


), 
new Zend_Form_Element_Button(submitButton', array( 

/省 略 代 码 用 于 设置 “提交 ”按钮 属性 
) 


3 
上 述 代码 中 的 Form_Bbs_Reply 类 继承 自 Zend_Form 类 ， 通 过 Zend_Form 类 中 定义 的 方法 来 为 
复 问题 表单 设置 属性 ， 并 通过 addElements() 方 法 将 表单 元 素 对 象 添加 到 问题 回复 表单 中 。 
(2) 用 户 输入 完成 回复 内 容 , 提交 表单 后 ， 所 有 回复 内 容 将 被 提交 到 QuestionController 控制 器 的 
threadAction 动作 中 进行 保存 ， 其 中 用 于 保存 用 户 回 复 信息 的 关键 代码 如 下 : 
倒 竹 26 ”代码 位 置 : 光盘 \TM\09\known\application\modules\default\controllers\QuestionController.php 


互 


if ($this->_request->isPost()) { 1/ 判断 用 户 是 否 提交 了 表单 
$formData = $this->_request->getPost(); // 获 取 表 单 提交 数据 所 组 成 的 数组 
if ($replyForm->isValid($formData)) { // 如 果 通 过 表单 验证 
S$replyid = $this->_request->getParam('replyid'); // 获 得 问题 类 别 ID 
if (isset($replyid) && $replyid != null) { // 如 果 设 置 了 问题 类 别 1D, 则 说 明 要 进行 更 改 问题 回复 


操作 ， 否 则 进行 添加 问题 回复 操作 
$updateArray = array('title' => $formData['title] , "content => $formData[' content]); 
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// 更 改 问题 回 复数 组 
$where = $this->_replyModel->getAdapter()->quotelnto('id = ?, $replyid); ”// 更 改 条 件 
S$this->_replyModel->update($updateArray, $where); // 更 改 数据 
S$this->_redirect('/question/thread/param/" . $titleid . '-' . $this->_request->getParam(‘topage') . gr . 
S$replyid); /| 重 定向 到 问题 详细 信息 页 面 
}else{ 


S$replyUser = $this->_userModel->findByNetname($this->_auth->getidentity());”// 获 取 回 复 人 信息 
SinsertArray = array('title' => $formDatar'title] , 'content' => $formData['content] , addtime' => date 
(Y-m-d Hii:s') , 'user_id' => $replyUserfid] , ‘ttle_id' => $arrayTitle[tid] , ‘topindex' => 0 , 'bbstype_id' => $arrayTitle 


[bbstype_id]); /添加 问题 回复 数组 
S$this->_replyModel->insert($insertArray); 1/ 添 加 数据 
StotalReply = $arrayTitle['totalreply] + 1; 
© StotalPage = ceil($totalReply / $pageSize); 
@ S$this->_redirect(/question/thread/param/" . $titleid . '-'. $totalPage . #b); 。 // 重 定向 到 问题 详细 信息 页 面 
} 
}else{ 
foreach ($replyForm->getMessages() as $messageArray) { // 获 取 错 误 信 息 
foreach ($messageArray as $message) { 
S$errorMsg .= $message . '<br />'; 
} 
} 
S$replyForm->populate($formData); // 表 单数 据 回 填 
$ 
} 
Ah 代码 贴 十 


@ 函数 ceil() 的 作用 是 对 浮 点 数 进行 向 上 取 整 。 
四 _redirect() 方 法 的 作用 是 使 当前 页 面 重 定向 到 其 他 页 面 。 


上 述 代码 首先 通过 当前 控制 器 对 象 的 _request 属性 的 isPost0 方 法 ， 判 断 用 户 是 否 提交 表单 ， 如 果 
是 则 首先 对 表单 提交 的 数据 进行 数据 校 验 , 如 果 成 功 通过 校 验 , 则 通过 回复 表 (tb_reply ) 模型 的 insert0 
方法 ， 将 回复 信息 保存 到 数据 库 中 。 


9.12 程序 调试 与 错误 处 理 


在 程序 开发 过 程 中 ， 难 免 会 遇 到 各 种 错误 ， 为 了 能 够 快速 地 解决 程序 开发 过 程 中 所 遇 到 的 错误 ， 
需要 开发 人 员 具 有 良好 的 的 编程 习惯 和 敏锐 的 观察 能 力 ， 并 不 断 地 培养 个 人 的 开发 经 验 。 本 节 将 对 程 
序 开发 和 运行 过 程 中 常见 错误 的 调试 和 处 理 方法 进行 讲解 。 


9.12.1 项 目 开发 过 程 中 常见 错误 及 解决 方法 


为 了 能 够 显示 项 目 开发 过 程 中 的 错误 提示 ,需要 开发 php.ini 文 件 的 display_startup_errors 和 display_ 
errors 项 ， 或 者 在 本 站 的 application.ini 文件 中 加 入 如 下 配置 来 使 项 目 在 开发 过 程 中 显示 错误 提示 。 


phpSettings.display_startup_errors = 1 
phpSettings.display_errors = 1 


他 
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这 里 需要 注意 , 在 项 目 投入 生产 后 , 需要 将 上 述 两 项 的 值 设 为 0, 这 样 在 运行 过 程 中 即使 出 现 错误 ， 
也 不 会 在 页 面 中 打印 出 错误 信息 ， 从 而 可 以 防止 网 站 的 漏洞 被 非法 用 户 发 现 ， 有 效 地 提高 网 站 的 安全 
性 。 开 启 上 述 错误 提示 信息 后 ， 在 项 目 开发 过 程 中 ， 一 旦 遇 到 错误 ， 开 发 人 员 可 以 根据 错误 提示 推断 
错误 所 在 并 予以 解决 。 

在 PHP 5 以 上 版 本 中 , 引入 了 异常 处 理 机 制 , 开发 人 员 通 过 异常 处 理 可 以 捕获 程序 运行 时 的 错误 。 
异常 处 理 中 主要 有 try、catch 和 throw 这 3 个 关键 字 ， 其 语法 结构 如 下 : 


try{ 

// 要 执行 的 代码 块 
}catch(Exception $e){ 

// 异 常 处 理 方式 ， 这 里 可 以 打印 错误 信息 或 使 用 throw 语句 抛 出 异常 
} 


用 户 要 执行 的 代码 块 编写 在 try/catch 代码 块 之 间 , 通过 catch 语句 指定 要 捕获 的 异常 , 并 通过 throw 
语句 抛 出 异常 。 


9.12.2” 自 定义 错误 页 面 


为 了 提高 页 面 的 友好 程度 , 隐藏 项 目 生产 环境 
下 的 错误 , 经 常会 在 项 目 中 制作 一 个 单独 的 用 于 显 ”exalt 
示 错 误 提示 的 页 面 。 明日 知道 网 站 的 错误 提示 页 面 
如 图 9.35 所 示 。 A 
使 用 Zend Framework 框架 制作 项 目 时 ， 有 一 
个 专门 的 错误 控制 器 ErrorController 用 来 显示 页 面 < 
错误 提示 , 开发 人 员 只 需 在 该 控制 器 中 定义 一 个 名 图 9.35” 自 定义 的 错误 提示 页 面 
为 errorAction 的 动作 即 可 。 明 日 知道 网 站 中 的 错 
误 控制 器 代码 如 下 : 
倒 程 27 ”代码 位 置 ， 光盘 \TM\09\known\application\modules\default\controllers\QuestionController.php 
class ErrorController extends Zend_Controller_Action 


八 对 不 起 ， 氏 要 浏览 的 页 不 存在 区 必 生 本 知 错误 ， 请 与 本 由 管理 及 联系 | 


public function errorAction () 
{ 
S$this->view->title = "明日 知道 网 站 错误 提示 "; // 页 面 标题 
$this->_helper->layout->disableLayout(); /去 掉 布 局 
} 
} 


定义 完成 上 述 错 误 控制 器 后 ， 还 需要 定义 errro.phtml 视图 ， 该 视图 用 来 显示 错误 提示 的 内 容 。 
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明日 知道 网 站 力求 打造 成 为 国内 知名 的 专业 软件 开发 技术 搜索 和 交流 平台 ， 通 过 该 平台 开发 人 员 
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和 编程 初学 者 可 以 查找 到 大 部 分 开发 和 学 习 过 程 中 可 能 会 遇 到 的 技术 难题 , 即使 没有 查找 到 预期 结果 ， 
还 可 以 通过 该 平台 将 问题 发 表 出 去 ， 这 样 就 可 以 与 其 他 网 友 探 讨 ， 最 终 将 问题 解决 。 

在 开发 过 程 中 ， 通 过 严谨 的 需求 分 析 、 功 能 分 析 和 数据 库 分 析 的 全 过 程 ， 并 充分 考虑 到 生产 环境 
下 的 升级 与 维护 等 后 期 工作 , 本 站 采用 Zend 公司 研发 的 MVC 框架 Zend Framework 进行 开发 。 在 开发 
过 程 中 ， 使 用 到 目前 Web 开发 领域 热门 技术 ， 如 在 线 编辑 器 、 图 片 验证 码 、 类 似 Google 搜索 条 的 下 
拉 关 键 字 提 示 列 表 、 关 键 字 分 词 查询 和 查询 结果 中 关键 字 描 红 输 出 等 。 

通过 学 习 本 章 内 容 ， 可 以 对 项 目的 完整 开发 流程 有 所 了 解 ， 并 理解 MVC 开发 模式 的 含义 。 技 术 
方面 , 不 仅 可 以 掌握 目前 Web 应 用 的 主流 模块 和 功能 的 制作 方法 ， 而 且 可 以 熟练 应 用 Zend Framework 
框架 进行 项 目 开发 。 如 果 能 够 潜心 学 习 本 站 内 容 ， 相 信 定 能 受益 匪 浅 ， 使 个 人 技术 水 平 有 显著 的 提高 。 


MOE 


电子 商务 网 站 


(Apache+PHP+ADODB+Smarty+Ajax+MYyYSQL 实现 ) 


随 着 20 世纪 PC 机 【个 人 计算 机 ) 的 发 展 和 互联 网 的 普及 ， 电 子 
商务 从 报 文 时 代 进 入 到 了 Internet 时 代 , 并 逐渐 被 大 众 所 了 解 和 接受 。 
电子 商务 (Electronic Commerce，EC) 是 日 前 发 展 较 快 的 一 种 商务 模 
式 。 迄 今 为 止 ， 不同 领域 的 人 对 EC 的 理解 各 有 不 同 。 简 单 地 说 ，EC 
是 一 种 基于 Internet， 利 用 计算 机 硬件 、 软 件 等 现 有 设备 和 协议 进行 
各 种 商务 活动 的 方式 。 

通过 阅读 本 章 ， 可 以 学 习 到 : 

HH 了 解 如 何 进行 系统 分 析 

MH 了 解数 据 库 设计 流程 

中， 熟悉 搭建 系统 架构 的 方法 

H 掌握 Smarty 模板 技术 的 应 用 

MH 掌握 ADODB 类 库 技术 的 应 用 

Mm 熟悉 Smarty 模板 和 ADODB 类 库 的 配置 方法 

MH 党 所 注册 即时 验证 的 实现 方法 

WI 掌握 简 单 的 树 形 菜单 的 实现 方法 

MY 掌握 购 物 和 车 的 实现 方法 

My 掌握 订单 的 处 理 方法 
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10.1 开发 背 和 及 


自 20 世纪 90 年 代 ， 互 联网 的 蓬勃 发 展 ， 为 企业 提供 了 一 个 全 新 的 机 遇 。 企 业 网 站 、 电 子 商务 成 
为 热门 话题 。 其 中 ， 电 子 商务 更 是 关系 到 经 济 结构 、 产 业 升 级 和 国家 整体 经 济 竞争 力 。 为 此 ， 我 国 已 
经 将 发 展 电子 商务 列 为 信息 化 建设 的 重要 内 容 ， 并 努力 创造 条 件 ， 积 极地 推进 电子 商务 的 发 展 。 

据 美国 在 线 (AOL) 和 Henley Centre 联合 进行 的 一 项 调查 显示 : 国外 有 80% 的 受 调查 者 会 选择 网 
上 购物 或 寻求 帮助 ，10% 的 受 调查 者 会 选择 熟悉 的 品牌 或 厂商 来 购买 。 而 在 国内 ， 自 1997 年 拉 开 了 电 
子 商务 的 序幕 ， 短 短 的 10 年 时 间 里 ， 全 国 已 有 4 万 家 商业 网 站 ， 几 乎 每 天 都 有 新 的 网 站 诞生 ， 厂 商 所 
在 地 也 从 上 海 、 广 州 、 深 圳 等 沿海 发 达 地 区 扩展 到 全 国 各 大 、 中 城市 。 


10.2 需求 分 析 


随 着 “地 球 村 ”概念 的 兴起 ， 网 络 已 经 深入 到 人 们 生活 的 每 一 个 角落 。 世 界 越 来 越 小 ， 信 息 的 传 
播 越 来 越 快 ， 内 容 也 越 来 越 丰富 。 现 在 ， 人 们 对 于 在 网 络 上 寻求 信息 和 服务 已 不 再 满足 于 简单 的 信息 
获取 上 ， 人 们 更 多 的 是 需要 在 网 上 实现 方便 的 、 便 捷 的 、 可 交互 式 的 网 络 服务 。 电 子 商务 则 正好 满足 
了 人 们 的 需求 。 它 可 以 让 人 们 在 网 上 实现 互动 的 交流 及 足 不 出 户 地 购买 产品 ， 向 企业 发 表 自 己 的 意见 、 
服务 需求 及 有 关 投 诉 ， 并 且 通 过 网 站 的 交互 式 操作 向 企业 进行 产品 的 咨询 、 得 到 相应 的 回馈 及 技术 支 
持 。 精 明 的 商家 绝 不 会 错过 这 样 庞大 的 市 场 ， 越 来 越 多 的 企业 已 经 开展 了 电子 商务 活动 。 加 入 电子 商 
务 的 行列 也 许 不 会 让 企业 马上 见 到 效益 ， 但 不 加 入 则 一 定 会 被 时 代 所 抛弃 。 


10.3 系统 分 析 


10.3.1 系统 目标 


根据 客户 提供 的 需求 和 对 实际 情况 的 考察 与 分 析 ， 该 电子 商务 应 该 具备 如 下 特点 : 
首页 设计 要 能 够 吸引 用 户 的 目光 ， 整 个 页 面 要 以 简洁 为 主 ， 突 出 重点 。 

可 操作 性 强 ， 避 人 免 复杂 的 、 有 异议 的 链接 。 

浏览 速度 快 ， 尽 量 避 免 长 时 间 打 不 开 页 面 的 情况 发 生 。 

商品 信息 部 分 有 实物 图 例 ， 图 像 清楚 、 文 字 醒 目 。 

详细 的 商品 查询 功能 ， 可 以 通过 商品 的 各 个 属性 来 搜索 。 

详细 的 流程 介绍 ， 从 浏览 商品 到 购买 结账 ， 各 个 步骤 之 间 的 联系 最 好 能 以 图 例 来 说 明 。 
提供 在 线 咨 询 。 

后 台 可 以 对 用 户 信息 和 商品 信息 进行 详尽 的 查看 和 管理 。 

订单 管理 。 

易 维 护 ， 并 提供 二 次 开发 支持 。 


名 
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10.3.2 ”系统 功能 结构 


电子 商务 平台 分 前 台 系 统 和 后 台 系 统 。 下 面 分 别 给 出 前 、 后 台 的 系统 功能 结构 图 。 电 子 商务 前 台 
系统 功能 结构 如 图 10.1 所 示 。 


电子 商务 前 台 系 统 功能 结构 


起 
丰 以 藻 卉 


芷 1 


涡 莹 营 可 
洲 芯 


查 
看 
网 
物 
库 


图 10.1 电子 商务 前 台 系 统 功能 结构 图 
电子 商务 后 台 系统 功能 结构 图 如 图 10.2 所 示 。 
电子 商务 后 台 系 统 功能 结构 图 


嫩 蚤 渤 洋 
总 苞 态 俏 


副 直 车 就 
长 马凯 时 
ae 


溢 酸 台阶 


一 | 蝶 啼 六 只 


修 | | 出 | 作出 | 全 出 | | 活 | [| jm | 修 | | 出 
改 | | 除 | 改 | | 除 | | 加 | | 除 | | 各 | 区 | | 星 | | 看 | | 除 | | 结 改 | | 除 
类 | | 类 || 奋 | 座 | | | | 入 | | 答 | 符 | | 会 | 会 | 会 链 | | 链 
别 | 品 || 昌 | | 曲 | | 单 | | 音 | 要 员 | | 品 | | 品 || 吕 搂 | | 接 


图 10.2 电子 商务 后 台 系 统 功 能 结构 图 
10.3.3 ”功能 预览 


电子 商务 网 站 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 儿 个 
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典型 功能 的 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 
鹏 逊 购物 系统 主页 如 图 10.3 所 示 ， 该 页 面 展示 网 站 的 部 分 最 新 商品 、 热 门 商品 、 推 荐 商品 以 及 网 
站 的 最 新 公告 和 会 员 登录 窗口 。 最 新 商品 展示 页 面 如 图 10.4 所 示 , 该 页 面 分 页 展示 网 站 的 所 有 最 新 商品 。 


天 砂 名 可 ， 降价 


SMT TN EN tN Ty 


ER Fo | ee | Iva Ta TaagR inPag Jag 1 
图 10.3 网 站 主页 面 图 10.4 最 新 商品 展示 


购物 车 页 面 如 图 10.5 所 示 ， 该 页 面 展示 会 员 在 本 站 购买 的 商品 。 
收银 台 页 面 如 图 10.6 所 示 ， 该 页 面 完成 收 货 人 信息 的 填写 ， 并 提交 订单 。 


商品 各 称 购买 数量 市 场 格 会 员 价 相 i 人 EPETTT Ma; |13005 
加 车 [| 可 ww | a lm | Er - 
网 | 和 RU 辽 | 区 |] 0 We | 8 S100 区 [5 
5 洗衣 机 刀 1000 ao | 日 zm | isK5x: [可 
万 | mt E 加 wm | 9 人 mR 
全 奸 后 地。 册 除 过 过 [EE EE 共计 11277 元 | 
图 10.5 购物 车 图 10.6 收银 台 


10.3.4 系统 流程 图 


为 了 便于 开发 人 员 了 解 系统 各 个 功能 模块 之 间 的 联系 及 完整 的 购物 流程 , 下 面 给 出 系统 的 流程 图 ， 
如 图 10.7 所 示 。 


10.3.5 ”开发 环境 


在 开发 电子 商务 平台 时 ， 该 项 目 使 用 的 软件 开发 环境 如 下 。 
1. 服务 器 端 


操作 系统 : Windows Server 2003/Linux (推荐 )。 
服务 器 : Apache 2.2.8。 


@ 
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3 浏览 其 他 商品 


图 10.7 系统 流程 图 


PHP 软件 : PHP 5.2.6。 

数据 库 : MySQL 5.0.51。 

MySQL 图 形 化 管理 软件 : phpMyAdmin-2.10.3。 
ADODB 类 库 。 

Smarty 模板 。 

开发 工具 : Dreamweaver 8。 

浏览 器 : IE 6.0 及 以 上 版 本 。 

分 辨 率 : 最 佳 效 果 为 1024x768 像素 。 


客户 端 


浏览 器 : 推荐 IE 6.0 及 以 上 版 本 。 
分 辨 率 ， 最 佳 效 果 为 1024x768 像素 。 


10.3.6 ”文件 夹 组 织 结构 


因 办 办 办 办 办 办 办 


R 


四 办 


编写 代码 之 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创建 出 来 (例如 ， 创 建 一 个 名 为 images 的 文件 
夹 , 用 于 保存 程序 中 所 使 用 的 图 片 ), 这样 不 但 可 以 方便 以 后 的 开发 工作 , 也 可 以 规范 系统 的 整体 架构 。 
因为 本 项 目 使 用 的 是 Smarty+ADODB 技术 ， 所 以 目录 较 多 。 下 面 介绍 一 下 本 系统 的 目录 结构 (到 三 级 
目录 )， 如 图 10.8 所 示 。 
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图 10.8 电子 商务 网 站 文件 夹 组 织 结 构图 
10.4 数据 库 设 计 


无 论 是 什么 系统 软件 ， 其 最 根本 的 功能 就 是 对 数据 的 操作 与 使 用 。 所 以 ， 一 定 要 先 做 好 数据 的 分 
析 、 设 计 与 实现 ， 然 后 再 实现 对 应 的 功能 模块 。 


10.4.1 数据 库 分析 


根据 需求 分 析 和 系统 的 功能 流程 图 , 找 出 需要 保存 的 信息 数据 (也 可 以 理解 为 现实 世界 中 的 实体 )， 
并 将 其 转化 为 原始 数据 《〈 属 性 类 型 ) 形式 。 这 种 描述 现实 世界 的 概念 模型 ， 可 以 使 用 E-R 图 来 表示 ， 
也 就 是 实体 -联系 图 。 最 后 将 E-R 图 转换 为 关系 数据 库 。 这 里 重点 介绍 几 个 E-R 图 。 

1. 会 员 信息 实体 


会 员 信息 实体 包括 编号 、 名 称 、 密 码 、E-mail、 身 份 证 号 、 固 定 电话 、QQ、 密 码 保护 、 密 码 答案 、 
邮编 、 注 册 日 期 、 真 实 姓名 等 属性 。 会 员 信息 实体 E-R 图 如 图 10.9 所 示 。 


图 10.9 会 员 信息 实体 E-R 图 
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2. 商品 信息 实体 
商品 信息 实体 包括 编号 、 名 称 、 介 绍 、 添 加 日 期 、 型号、 图 片 、 库 存 、 销 售 量 、 类 型 、 会 员 价格 、 
市 场 价格 、 是 否 新 品 等 属性 。 商 品 信息 实体 E-R 图 如 图 10.10 所 示 。 


图 10.10 商品 信息 实体 E-R 图 


除 上 面 介绍 的 2 个 E-R 图 ， 还 有 商品 订单 实体 、 商 品评 价 实体 、 公 告 实体 、 管 理 员 实体 、 类 型 实 
体 和 友情 链接 实体 等 ， 限 于 篇 幅 ， 这 里 仅 列 出 主要 的 实体 E-R 图 。 


10.4.2 ”创建 数据 库 和 数据 表 


系统 E-R 图 设计 完成 后 ， 接 下 来 根据 E-R 图 来 创建 数据 库 
和 数据 表 。 首 先 来 看 一 下 电子 商务 平台 所 使 用 的 数据 表情 况 ， 如 。 Ps Dn 


图 10.11 所 示 。 ME A at 

下 面 来 看 各 个 数据 表 的 结构 和 字段 说 明 。 Te 
ri ER 

tb admin (管理 员 信息 表 ) en A 


也 -public 。 MySAM gb2312_chinese_cl 。。 公 结 信息 表 


管理 员 信息 表 主 要 用 于 存储 管理 员 的 信息 ,其 结构 如 图 10.12 Im ww maa mana esa 
所 示 。 图 10.11 电子 商务 数据 表 
tb_class《〈 商 品类 型 表 ) 
商品 类 型 表 主要 用 于 添加 商品 的 类 别 ， 可 以 设 定 多 个 子 类 别 〈 目 前 最 多 只 能 到 二 级 子 类 别 )， 其 结 
构 如 图 10.13 所 示 。 


图 服务 器 : localhost ) 避 数据 库 . db_shop 上 国 表 : tb_admin 加 服务 秋 - localhost > 总 数据 库 - db shop 上 匡 才 :th_class 

字段 类 型 整理 区 性 Null 默认 预 外 说 明 定 民 美 型 整理 区 性 Null 著 认 莉 外 说 明 

志 Int(q) 百 aUko-incrament 电动 篇 号 过 int(4) 至 autc_incremant 生动 旺 号 

name VarcharEoj gb2312.chineseci 再 name varcharfzo) gb2312.chinese_ci 否 类 型 名 狂 

pwd 。 varcharl2ol 曲 2312_chirese_ci 在 ee Mevel ina 妊 1 sg 
图 10.12 管理 员 信息 表 结 构 图 10.13 商品 类 型 表 结构 


tb_ commo《〈 商 品 信息 表 ) 

商品 信息 表 主 要 用 于 存储 关于 商品 的 相关 信息 ， 其 结构 如 图 10.14 所 示 。 

此 外 还 有 商品 订单 表 、 商 品评 论 表 、 公 告 信 息 表 、 友 情 链 接 表 和 会 员 信息 表 ， 限 于 篇 幅 ， 这 里 不 
再 介绍 ， 读 者 可 参见 本 书 附 赠 光 盘 中 的 数据 库 文件 。 
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加 服务 天 localhost ) 加 数据 库 - db_shop ;回去 - th_commo 
全 3 可 下 县 Null 里 新 # 说 明 
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a 


10.14 ”商品 信息 表 结 构 
10.5 公共 文件 设计 


公共 模块 就 是 将 多 个 页 面 都 可 能 使 用 到 的 代码 写成 单独 的 文件 , 在 使 用 时 只 要 用 include 或 require 
语句 将 文件 包含 进来 即 可 ， 如 本 系统 中 的 数据 库 连 接 、 管 理 和 分 页 类 文件 ，Smarty 模板 配置 类 文件 ， 
类 的 实例 化 文件 ，CSS 样式 表 文 件 ，js 脚本 文件 等 。 以 前 台 系统 为 例 ， 下 面 给 出 主要 的 公共 文件 ， 后 
台 的 公共 文件 与 前 台大 同 小 异 。 


10.5.1 数据 库 连 接 、 管 理 和 分 页 类 文件 


在 数据 库 连 接 、 管 理 和 分 页 类 文件 中 ,定义 3 个 类 ,分 别 是 ConDB 数据 库 连 接 类 ,实现 通过 ADODB 
连接 MySQL 数据 库 ; AdminDB 数据 库 管理 类 ， 使 用 ADODB 类 库 中 的 方法 执行 对 数据 库 中 数据 的 查 
询 、 添 加 、 更 新 和 删除 操作 ; SepPage 分 页 类 ， 用 于 对 商城 中 的 数据 进行 分 页 输出 。 

倒 程 01 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\system\system.class.inc.php 


<?php 
/| 数据 库 连接 类 
class ConnDB{ 
var $dbtype; 
var $host; 
var $user; 
var $pwd; 
var $dbname; 
var $debug; 
var $conn; 
function ConnDB($dbtype,$host, $user,$pwd,$dbname,$debug=false){ /构造 方法 ， 为 成 员 变 量 赋值 
$this->dbtype=$dbtype; 
S$this->host=$host; 
S$this->user=$user; 
$this->pwd=$pwd; 
$this->dbname=$dbname; 
S$this->debug=$debug; 
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} 
function GetConnld(X{ 
require("adodb/adodb.inc.php"); 
if($this->dbtype=="mysql")f 


$this->conn=NewADOConnection("mysql"); 


// 实 现 与 不 同 数据 库 的 连接 并 返回 连接 对 象 
/调用 ADODB 类 库 文 件 

// 判 断 成 员 变 量 传 递 的 数据 库 类 型 

/| 执行 与 MySQL 数据 库 的 连接 


S$this->conn->Connect($this->host,$this->user,$this->pwd,$this->dbname); // 用 户 、 密 码 


S$this->conn->Execute("set names gb2312"); 


if($this->dbtype=="mysql") 
$this->conn->debug=$this->debug; 
return $this->conn; 


上 
function CloseConnld(X{ 
$this->conn->Disconnect(); 


} 


} 
/数据 库 管理 类 
class AdminDB{ 
function ExecSQL($sqlstr,$conn){ 
$sqltype=strtolower(substr(trim($sqlstr),0,6)); 
$rs=$conn->Execute($sqlstr); 
if($sqltype=="select"}{ 
$array=$rs->GetRows(); 
if(count($array)==0 || $rs==false) 
return false; 
else 
return $array; 


// 设 置 数据 库 的 编码 格式 


// 返 回 连接 对 象 


/定义 关闭 数据 库 的 方法 
/执行 关闭 的 操作 


/定义 方法 , 参数 为 SQL 语句 和 连接 数据 库 返 回 的 对 象 
// 截 取 SQL 中 的 前 6 个 字符 串 ， 并 转换 成 小 写 

/执行 SQL 语句 

// 判 断 如果 SQL 语句 的 类 型 为 SELECT 

/执行 该 语句 ， 获 取 查 询 结果 

// 判 断 语 句 是 否 执行 成 功 

// 如 果 查 询 结果 为 0， 或 者 执行 失败 ， 则 返回 false 


/否则 返回 查询 结果 的 数组 


}elseif ($sqltype=="update" || $sqltype=="insert" || $sqltype=="delete"){ 
// 判 断 如 果 SQL 语句 类 型 不 为 SELECT， 则 执行 如 下 操作 


if($rs) 

retum true; 
else 

returmn false; 


} 


} 

// 分 页 类 

class SepPage{ 
var $rs; 
var $pagesize; 
var $nowpage; 
var $nowpages; 
var $array; 
var $conn; 
var $sqlstr; 


/执行 成 功 返 回 true 


/否则 返回 false 


function ShowDate($sqlstr,$conn,$pagesize, $nowpage){ /定义 方法 


if(!isset($nowpage) || Snowpage=="") 
S$this->nowpage=10; 
else 


// 判 断 变 量 值 是 否 为 空 
/定义 每 页 输出 的 记录 数 
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Sthis->nowpage=$nowpage; 


$this->pagesize=$pagesize; /定义 每 页 输出 的 记录 数 
S$this->conn=$conn; /| 连接 数据 库 返 回 的 标识 
S$this->sqlstr=$sqlstr; /| 执行 的 查询 语句 
Sthis->rs=$this->conn->PageExecute($this->sqlstr,$this->pagesize,$this->nowpage); 
@$this->array=$this->rs->GetRows\(); /| 获取 记 录 数 

if(count($this->array)==0 || $this->rs==false) 

return false; 
else 


return $this->array; 
上 
function ShowPage($contentname,$utits,$anothersearchstr$class,$pagejf 
$allrs=$this->conn->Execute($this->sqlstr); /执行 查询 语句 
$record=count($allrs->GetRows()); /统计 记录 总 数 
$pagecount=ceil($record/$this->pagesize); 。“// 计 算 共 有 几 页 
$str.=" 共 有 ".$contentname."&nbsp;".$record."&nbsp;".$utits."&nbsp; 每 页 显示 &nbsp;".$this-> pagesize. 
"&nbsp;".$utits."&nbsp; 第 &nbsp;".$this->rs->AbsolutePage()."&nbsp; 页 / 共 &nbsp;".$pagecount."&nbsp; 页 "; 
$str.="&nbsp;&nbsp;&nbsp;&nbsp;"; 
if(!$this->rs->AtFirstPage()) 
$str="<a href=".$_ SERVER[PHP_SELF]."?page=".$page."&pages=1".$anothersearchstr” class= 
".$class."> 首 页 </a>"; 
else 
$str.="<font color=#555555'> 首 页 </font>"; 
$str.="&nbsp;"; 
if(!$this->rs->AtFirstPage()) 
$str="<a href=".$_ SERVER[PHP_SELF]"?page="$page."&pages="($this-> rs->AbsolutePage()-1). 
$anothersearchstr." class=".$class."> 上 一 页 </a>"; 
else 
$str.="<font color=#555555'> 上 一 页 </font>"; 
$str.="&nbsp;"; 
if(!$this->rs->AtLastPage()) 
S$str.="<a href=".$_SERVER[PHP_SELF']."?page=".$page."&pages=".($this->rs-> AbsolutePage()+1). 
$anothersearchstr." class=".$class."> 下 一 页 </a>"; 
else 
$str.="<font color=#555555'> 下 一 页 </font>"; 
$str.="&nbsp;"; 
if(!$this->rs->AtLastPage()) 
$str="<a href=".$_SERVER[PHP_SELF]."?page=".$page."&pages=".$pagecount. $anothersearchstr" 
class=".$class."> 尾 页 </a>"; 
else 
$str="<font color=#555555'> 尾 页 </font>"; 
if(count($this->array)==0 || $this->rs==false) 
return ™"; 
else 
return $str; 
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10.5.2 Smarty 模板 配置 类 文件 


在 Smarty 模板 配置 类 文件 中 配置 Smarty 模板 文件 、 临 时 文件 、 配 置 文件 等 文件 路 径 。 
system.smarty.inc.php 文件 的 代码 如 下 : 
全 性 02 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\system\system.smarty.inc.php 


<?php 

require("smarty/Smarty.class.php"); // 调 用 Smarty 类 文件 

class SmartyProject extends ”Smarty{ /定义 类 ， 继 承 Smarty 父 类 

function SmartyProject(X{ /定义 方法 ， 配 置 Smarty 模板 

$this->template dir = "./"; /| 指定 模板 文件 存储 在 根 目录 下 
$this->compile_dir = "./system/templates_c/";”// 指 定编 译文 件 存储 位 置 
S$this->config_dir = "./system/configs/"; /指定 配置 文件 存储 位 置 
S$this->cache_dir = "/systerycache/"; /指定 缓存 文件 存储 位 置 

?> 


10.5.3 ”类 的 实例 化 文件 


在 system.inc.php 文件 中 ,通过 require 语句 包含 system.smarty.inc.php 和 system.class.inc.php 文件 ， 
执行 类 的 实例 化 操作 ， 并 定义 返回 对 象 。 完 成 数据 库 连 接 类 的 实例 化 后 ， 调 用 其 中 的 GetConnld() 方 法 
连接 数据 库 。system.inc.php 文件 的 代码 如 下 : 

倒 程 03 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\system\system.inc.php 

<?php 

require("system.smarty.inc.php"); 

require("system.class.inc.php"); 

$connobj=new ConnDB("mysql","localhost","root","root","db_shop",false); // 数 据 库 连 接 类 实例 化 


$conn=$connobj->GetConnld(); /| 连接 数据 库 并 返回 连接 标识 
$admindb=new AdminDB(); /| 数据 库 操作 类 实例 化 
$seppage=new SepPage(); // 分 页 类 实例 化 

$smarty=new SmartyProject(); /Smarty 模板 配置 类 的 实例 化 
?> 


10.6 前 台 首 页 设计 
10.6.1 前 台 首 页 概述 
前 台 首 页 一 般 没有 多 少 实质 的 技术 ， 主 要 是 加 载 一 些 功 能 模块 ， 如 登录 模块 、 导 航 栏 模块 、 公 告 


栏 模块 等 ， 使 浏览 者 能 够 了 解 网 站 的 内 容 和 特点 。 首 页 的 重要 之 处 是 要 合理 地 对 页 面 进行 布局 ， 既 要 
尽 可 能 地 将 重点 模块 显示 出 来 ， 同 时 又 不 能 因为 页 面 凌乱 无 序 ， 而 让 浏览 者 无 所 适 从 、 产 生 反 感 。 本 
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系统 前 台 首 页 的 运行 结果 如 图 10.15 所 示 。 


图 10.15 前 台 首 页 运行 结果 
10.6.2 ”前 台 首 页 技术 分 析 


在 前 台 首 页 中 应 用 switch 语句 与 Smarty 模板 中 的 include 标签 设计 一 个 框架 页 面 ， 实 现 不 同 功 能 
模块 在 首页 中 的 展示 。 

switch 语句 在 PHP 动态 文件 中 使 用 ， 根 据 超 链 接 传递 的 值 ， 包 含 不 同 的 功能 模块 。 

include 标签 在 Smarty 模板 页 中 使 用 ， 在 当前 模板 页 中 包含 其 他 模板 文件 。 其 语法 如 下 : 


{include file="file_name " assign=" " var=” 个 


参数 说 明 : file 指定 包含 模板 文件 的 名 称 ;，assign 指定 一 个 变量 保存 包含 模板 的 输出 ，var 传递 给 
待 包含 模板 的 本 地 参数 ， 只 在 待 包含 模板 中 有 效 。 


10.6.3 ”前 台 首页 实现 过 程 
国 ” 前台 首页 使 用 的 数据 表 : tb_ commo、tb_public、tb_user、tb_links 
(1) 创建 index.php 动态 页 。 在 index.php 动态 页 中 ， 应 用 include_once() 语 句 包含 相应 的 文件 ， 应 
用 switch 语句 ， 以 超 链接 中 参数 page 传递 的 值 为 条 件 进 行 判断 ， 实 现在 不 同 页 面 之 间 跳 转 。index.php 
的 关键 代码 如 下 : 
倒 牌 04 ”代码 位 置 : 光盘 \TM\10\ShoppingSystem \index.php 


<?php 

session_start(); /| 初始 化 session 变量 
require_once("system/system.inc.php"); // 获 取 数 据 库 连 接 、 管 理 和 模板 类 返回 的 对 象 
include_once "public.php"; // 包 含 公告 模块 

include_once "links.php"; // 包 含 友情 链接 模块 


@ 
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// 判 断 session 变量 的 值 是 否 为 空 


if($_SESSION["member]==""){ 
/如 果 为 空 ， 则 为 模板 变量 赋值 为 F 


$smarty->assign("member","F"); 
}else{ 
$smarty->assign("member",$_SESSION[member]); /否则 将 session 变量 的 值 赋 给 模板 变量 
// 判 断 变量 page 的 值 是 否 为 空 


} 
if($_GETI"page"]=="" 
// 如 果 为 空 ， 为 模板 变量 赋值 为 F 


$smarty->assign("page","F"); 
} 
switch($_GET["page"]X{ /应 用 switch 语句 ， 根 据 条 件 进行 判断 ， 实 现 页 面 跳 转 
case "hyzx": 
include_once "member.php"; // 包 含 指定 的 PHP 文件 
if($_GET[action] == 'modify’}{ /根据 action 的 值 ， 指 定 不 同 的 模板 页 
$smarty->assign("switchs",'modifypwd.html");”// 将 指定 的 模板 页 赋 给 模板 变量 


}else{ 
$smarty->assign("switchs",'membershow.html'); 


break; 
case 'allpub': 
include_once 'allpub.php'; 
$smarty->assign("switchs","allpub.htmI"); // 将 指定 的 模板 页 赋 给 模板 变量 
break; 
…// 省 略 了 部 分 代码 
default: 
include_once "nominate.php'; 
$smarty->assign("switches", "nominate.html"); 
include_once 'newhot.php'; 
$smarty->assign("switchs","newhot.html"); 


} 
$smarty->display(index.html); /指定 模板 页 
?> 
(2) 创建 index.html 模板 页 。 在 模板 文件 index.html 中 应 用 Smarty 的 include 标签 调用 不 同 的 模 
板 文件 ， 生 成 静态 页 面 。 关 键 代 人 码 如 下 : 
倒 程 05 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\index.html 
<table width="850" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td colspan="2">{include file="top.html"}</td> 
</tr> 
<tr> 
<td width="216" align="left" valign="top"> 
{if $member=="F"} 
{include file="login.html 
{else} 
{include file="info.html"} 
{/ 诊 
{include file="public.html"} 
{include file="links.html"} 


</td> 
<td width="634" height="720" align="center" valign="top"> 
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{include file="search.html"} 
{include file=$switchs} 
{if $page==F} 
{include file=$switches} 
{if</td> 
</tr> 
</table> 
<table width="850" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td>{include file=buttom.html}</td> 
</tr> 
</table> 


CS 


本 系统 的 功能 较 多 ， 结 构 比 较 复杂 ， 对 于 初学 者 来 说 学 起 来 可 能 会 比较 困难 。 所以， 本 
书 将 系统 中 的 各 个 功能 模块 所 涉及 的 文件 (如 PHP、TPL、CSS、JS 等 ) 尽 可 能 都 单独 实现 。 读 者 
在 学 习 其 中 某 个 模块 时 ， 可 以 将 相关 的 文件 统一 放 到 同一 个 目录 下 单独 测试 。 


10.7 登录 模块 设计 


10.7.1 登录 模块 概述 


用 户 登 录 模 块 是 会 员 功 能 的 窗口 。 匿 名 用 户 虽 然 也 可 以 访问 本 网 站 ， 但 只 能 进行 浏览 、 查 询 等 简 
单 操作 ， 而 会 员 则 可 以 购买 商品 ， 并且 能 享受 超 低 价格 。 登 录 模 块 包括 


用 户 注册 、 用 户 登 录 和 找 回 密码 3 部 分 ， 其 运行 结果 如 图 10.16 所 示 。 oa | 
10.7.2 ”登录 模块 技术 分 析 I 


图 10.16 登录 模块 运行 结果 
(1) Ajax 技术 无 刷新 验证 用 户 名 是 否 被 占用 。 关 键 代码 如 下 : 


倒 程 06 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\check.js 


/*form 为 传 入 的 表单 名 称 ， 本 上段 代码 为 register 表单 */ 
function chkname(form){ 
/如 果 name 文本 域 的 信息 为 空 ， 则 名 为 name1 的 div 标签 显示 如 下 信息 */ 
if(form.name.value=="™"){ 
name1.innerHTML="<font color=#FF0000> 请 输入 用 户 名 ! </font>"; 
jelse{ 
/否则 获取 文本 域 的 值 */ 
var user = form.name.value; 
/* 生 成 url 链接 ， 将 user 的 值 传 到 chkname.php 页 进行 判断 */ 
var url = "chkname.php?user="+user; 
/使 用 xmlhttprequest 技术 运行 页 面 */ 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(X{ 


第 10 章 电子 商务 网 站 ( Apache+PHP+ADODB+Smarty+Ajax+MySQL 实现 ) 


if(xmlhttp.readyState == 4X{ 

/* 根 据 不 同 的 返回 值 ， 在 div 标签 中 输出 不 同 信息 */ 

var msg = xmlhttp.responseText; 

if(msg == '3'){ 
name1.innerHTML="<font color=#FF0000> 用 户 名 被 占用 ! </font>"; 
retum false; 

}else if(msg == '2){ 
name1.innerHTML="<font color=green> 燕 喜 您 ， 可 以 注册 !</font>"; 

让 如 果 用 户 名 正确 ， 则 将 隐藏 域 的 值 改 为 yes*/ 
form.c_name.value = "yes"; 

Jelse{ 
name1.innerHTML="<font color=green> 未 知 错误 </font>"; 

} 

} 


} 
xmlhttp.send(null); 


- 

在 该 函数 中 调用 chkname.php 页 ， 该 页 在 会 员 登 录 时 也 会 被 调用 ， 所 以 这 里 分 两 种 情况 ， 即 有 密 
码 和 无 密码 。 无 密码 为 注册 验证 ， 当 没有 返回 结果 时 ， 说 明 该 用 户 名 可 用 ;而 有 密码 为 登录 验证 ， 和 
无 密码 相反 , 只 有 查询 记录 存在 时 , 才 允 许 登录 , 并 将 用 户 名 和 用 户 ID 存储 到 session 中 。chkname.php 
页 面 的 代码 如 下 : 

倒 程 07 代码 位 置 ， 光 盘 \TM\10\ShoppingSystem\ chkname.php 


<?php 

session_start() /开启 session 支持 
include_once ‘system/system.inc.php’; // 加 载 数据 库 连 接 文件 
S$reback = '0'; // 声 明 返回 变量 ， 初 值 为 0 


$sql = "select * from tb_user where name=".$_GET[usen]."";， // 生 成 SQL 语句 
/下 面 的 SQL 语句 加 上 了 对 密码 的 比较 ， 这 是 用 来 验证 登录 的 ， 因 为 登录 验证 使 用 的 也 是 本 页 面 */ 
$password = $_GET[password']; 


if(lempty($password)){ 

$sql .=" and password = ".md5($password).""; // 如 果 密 码 不 为 空 ， 说 明 是 登录 验证 
} 
/执行 SQL 语句 */ 
Srst = $conn->Execute($sql) or die(execute error'’); /根据 name 字段 查询 数据 库 
if($rst->RecordCount() == 1 /| 判断 name 是 否 被 使 用 

Sreback = '3'; // 被 使 用 返回 3 
/这 里 也 是 对 登录 页 面 进行 的 操作 ， 当 用 户 名 和 密码 输入 正确 时 ， 将 用 户 名 和 1D 存 进 session 中 */ 
if(lempty($password)){ 


$_SESSION[member] = S$rst->fields['name'; 
$_SESSION[id] = $rst->fields[id]; 
Sreback = "1'; 
} 
}else{ 
S$reback = '2'; /未 被 使 用 返回 2 


echo $reback; 
?> 
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(2) GD2 函数 库 生 成 验证 码 ， 关 键 代码 如 下 : 
倒 各 08 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\ yzm.php 


<?php 
srand((double)microtime()*1000000); 
$im=imagecreate(42,16); 
$black=imagecolorallocate($im,0,0,0); 
$white=imagecolorallocate($im,255,255,255); 
$gray=imagecolorallocate($im,200,200,200); 
imagefill($im,0,0,$gray); 
for($i=0; $i<4;$i++X{ 
$str=mt_rand(1,3); 
$size=mt_rand(3,6); 
$authnum=substr($_GETInum],$i,1); 
imagestring($im, $size,(2+$i*10),$str,$authnum,imagecolorallocate($im,rand(0,130),rand(0,130),rand(0,130))); 
让 
for($i=0; $i<200;$i++X{ 
S$randcolor=imagecolorallocate($im,rand(0,255),rand(0,255),rand(0,255)); 
imagesetpixel($im,rand()%70,rand()%30,$randcolor); 
} 
imagepng($im); 
imagedestroy($im); 
?> 


10.7.3 用 户 注册 


+ 要 要 侣 ,可 注册 1 
ee] 


国 ” ”用户 注册 使 用 的 数据 表 : tb_user 

用 户 注 册页 面 的 主要 功能 是 新 用 户 注册 。 如 果 信 息 输入 
完整 而 且 符合 要 求 , 则 系统 会 将 该 用 户 信息 保存 到 数据 库 中 ， 
否则 显示 错误 原因 ， 以 便 用 户 改正 。 用 户 注册 页 面 的 运行 结 
果 如 图 10.17 所 示 。 

(1) 创建 registerhtml 模板 文件 ， 编 写 用 户 注册 页 面 。 
其 中 包含 createxmlhttp.js 和 check.js 两 个 JS 脚本 文件 .其 中 ， 
createxmlhttp.js 是 Ajax 的 实例 化 文件 ， 而 checkjs 对 用 户 注 图 10.17 用 户 注册 
册 信息 进行 验证 ， 并 且 返 回 验 证 结果 。 

(2) 创建 registerphp 动态 PHP 文件 ， 加 载 模板 。registerphp 文件 的 代码 如 下 : 

倒 程 09 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\registerphp 


<?php 
include_once 'system/system.inc.php'; // 载 入 配置 文件 
$smarty->assign('title', 新 用 户 注册 '); // 设 置 ttle 
$smarty->display(registerhtml'); /显示 registerhtml 
3 


(3) 创建 reg_chk.php 文件 , 获取 表单 中 提交 的 数据 ,将 数据 存储 到 指定 的 数据 表 中 。reg_chk.php 
文件 的 代码 如 下 : 


@ 
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倒 程 10 ”代码 位 置 : 光盘 \TM\10\ShoppingSystem\reg_chk.php 


<?php 
session_start(); /开启 session 支持 
include_once 'system/system.inc.php'; // 加 载 数据 库 连 接 文件 
获取 表单 信息 */ 
$name = $_POST[name'; 
$password = md5($_POST[pwd11); /使 用 md5 给 密码 加 密 


$question = $_POST[question]; 
$answer = $_ POST[answer]; 
$realname = $_POST[realname'; 
$card = $_POST[T'card']; 
Stel = $_POSTItel]; 
$phone = $_POST[phone]; 
$Email = $_POST[email]; 
$QQ =$_POST[qq1]; 
$code = $_POST[code'; 
$address = $_POST['address'; 
$addtime = $conn->DBDate(time!()); // 使 用 DBDate() 函 数 生成 当前 时 间 
J 
/* 生 成 SQL 语句 */ 
$sql = "insert into tb_user( name, password, question, answer realname, card, tel, phone, Email QQ, code, 
address, addtime ) "; 
$sql .= "values ($name', '$password','$question','$answer, '$realname', $card', '$tel', '$phone'’, '$Email', '$QQ', 
'$code', '$address',$addtime)"; 
让 执行 SQL 语句 */ 
Srst = $conn->execute($sql); 
if($rst == false){ 
echo '<script>alert(\ 添 加 失败 \);history back;</script>'; 
jelse{ 
$_SESSION[member] = $name; 
$_SESSION[id] = $conn->Insert_ID(); 
echo "<script>top.opener.location.reload();alert(' 注 册 成 功 ');window.close();</script>"; 
} 


2% 
(4) 创建 “用 户 注册 ” 超 链 接 。 当 用 户 单 击 前 台 的 革 护 钮 时 ， 系 统 会 调用 js 文件 的 onClick 
事件 ， 弹 出 注册 窗口 。 代 码 如 下 : 
倒 竹 11 代码 位 置 : 光盘 \TM\10\ShoppingSystem\login.html 
<a href="#" id="login" onClick="reg()"> 用 户 注册 </a> 


这 里 使 用 到 的 js 文件 为 jyloginjs， 调 用 的 函数 为 reg()。 该 函数 的 代码 如 下 : 
倒 程 12 代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\login.js 


function reg(){ 
window.open("registerphp", "_blank", "width=500,height=450",false); /弹出 新 窗口 
} 
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10.7.4 ”用 户 登录 


国 ” 用户 注册 使 用 的 数据 表 : tb_user 
用 户 登 录 模块 的 运行 结果 如 图 10.18 所 示 ， 需 要 输入 用 户 名 、 密 码 和 验 
证 码 。 
(1) 创建 模板 文件 login.html， 完 成 用 户 登 录 表 单 的 设计 。 在 该 页 面 中 
当 单 击 Submit 按钮 时 ， 系 统 将 调用 lg(O) 函 数 对 用 户 登 录 提 交 信 息 进行 验证 。 
lg(0) 函 数 包含 在 js/login.js 脚本 文件 内 ， 代 码 如 下 : 
倒 姓 13 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\login.js 


function lg(form){ 
检验 用 户 名 */ 
if(form.name.value=="™"){ 
alert(' 请 输入 用 户 名 '); 
form.name.focus(); 
return false; 


} 

让 检验 密码 */ 

if(form.password.value == "" || form.password.value.length < 6X{ 
alert(' 请 输入 正确 密码 '); 
form.password.focus(); 
return false; 


} 

/检验 验证 码 ”/ 

if(form.check.value == ""){ 
alert(' 请 输入 验证 码 '); 
form.check.focus(); 
return false; 


} 

上 比较 验证 码 

if(form.check.value != form.check2.value){ 
form.check.select(); 
code(form); 
return false; 


} 
/根据 输入 的 用 户 名 和 密码 生成 url*/ 
var user = form.name.value; 
var password = form.password.value; 
var url = "chkname.php?user="+user+"&password="+password; 
/使 用 xmlhttp 对 象 来 返回 验证 结果 */ 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4){ 
var msg = xmlhttp.response Text; 
广 如 果 返 回 值 不 等 于 2， 说 阴 用 户 名 或 密码 错误 */ 
if(msg != '2°'}{ 
alert(' 用 户 名 或 密码 错误 路 ); 
form.password.select(); 


@ 


图 10.18 ”用户 登录 页 面 
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form.check.value = "; 
code(form); 


Jelse{ 

让 如 果 成 功 ， 重 新 刷新 本 页 */ 
alert(' 欢 迎 光临 "); 
location.reload(); 


xmlhttp.send(null); 
retur false; 


} 


用 户 名 和 密码 是 在 chkname.php 页 面 中 被 验证 ,chkname.php 在 10.7.2 节 中 已 经 介绍 , 这 里 不 再 重复 。 
(2) 创建 用 户 信息 模板 文件 info.html。 用 户 登 录 成 功 后 ， 在 原 登 录 框 位 置 将 显示 用 户 信息 ， 用 户 
可 以 通过 “会 员 中 心 ” 对 自己 的 信息 做 修改 ， 也 可 以 单 击 “ 查 看 购物 车 ” 超 链 接 查 看 购物 车 商品 ， 当 
用 户 离开 时 可 以 单 击 “ 安 全 离开 ” 超 链 接 。 用 户 信息 模块 的 主要 代码 如 下 : 
倒 性 14 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\info.html 


<!-- 显 示 当 前 登录 用 户 名 --> 
欢迎 您 ， {$member} 
<!-- 会 员 中 心 超 链接 --> 


<a href="?page=hyzx" id="info" class="lk"> 会 员 中心 </a> 
<!-- 查 看 购物 车 --> 


<a href="?page=shopcar" class="lk"> 查 看 购物 车 </a> 
<!- 安 全 离开 -> 


<a onclick="javascript:logout()" style="cursor:hand" id="info"> 安 全 离开 </a> 


10.7.5” 找 回 密码 


国 ”用 户 注册 使 用 的 数据 表 : tb_user 

登录 模块 的 最 后 一 个 部 分 就 是 找 回 密码 。 找 回 密码 是 根据 用 户 在 填写 资料 时 所 填写 的 密 保 问 题 和 
密 保 答案 来 实现 的 。 当 用 户 单 击 “ 找 回 密 码 ” 超 链接 时 ， 首 先 提 示 用 户 输入 要 找 回 密 码 的 会 员 名 称 ， 
然后 根据 密 保 问题 填写 密 保 答案 ， 最 后 重新 输入 密码 。 找 回 密码 模块 的 流程 如 图 10.19 所 示 。 


第 一 步 : 输入 会 员 名 称 第 二 步 : 输入 密 保 答案 


第 三 步 : 更 新 密码 第 四 步 : 更 改 成 功 
图 10.19 找 回 密码 流程 图 
1. 创建 模板 文件 
虽然 找 


回 密码 需要 4 个 步骤 ， 但 实际 上 每 个 步骤 使 用 的 都 是 相同 的 模板 文件 和 js 文件 ， 只 是 被 调 
用 的 表单 和 js 函数 略 有 差别 。 这 里 根据 不 同 的 文件 来 分 别 进行 介绍 。 
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该 模板 文件 一 共 包含 了 3 个 表单 ， 分 别 代表 3 个 步骤 ， 其 核心 代码 如 下 : 
倒 程 15 ”代码 位 置 ， 光盘 \TMN10\ShoppingSystem\found.html 
<!-- 载 入 两 个 js 脚本 文件 --> 


<script language="javascript" src="js/createxmlhttp.js"></script> 
<script language="javascript" src="js/found.js"></script> 
<!-- 第 1 个 div 标签 -> 
<div id= "first"> 
<table width="200" border="0" cellspacing="0" cellpadding="0"> 
<form id="foundname" name="found" method="post" action="#"> 
<tr><td> 找 回 密码 </td></tr> 
<tr><td> 会 员 名 称 : </td> 
<!--text 文本 域 ， 用 于 输入 要 找 回 密码 的 会 员 名 称 --> 
<td><input id="user" name="user type="text" class="txt"></td> 
</tr> 
<tr><td> 
<!-- 单 击 “ 下 一 步 ” 按 钮 ， 能 触发 onClick 事件 来 调用 chkname() 函 数 --> 
<input id = " next1 " name = " next1 " type = " button " class = " btn " value = " 下 一 步 "onClick = " return 
chkname ( foundname ) "/></td></tr> 
</form> 
</table> 
</div> 
<!-- 第 2 个 div 标签 ， 样 式 为 隐藏 -> 
<div id="second" style="display:none;"> 
<table> 
<form id="foundanswer" name="found" method="post" action="#"> 
<tr><td > 找 回 密码 </td></tr> 
<tr><td> 密 保 问题 : </td> 
<!-- 用 于 显示 密 保 问题 的 div 标签 -> 
<td <div id="question"></div></td></tr> 
<tr><td> 密 保 答案 : </td> 
<!-- 文 本 域 ， 用 于 填写 密 保 答案 -> 
<td ><input id="answer" name="answer" type="text" class="txt" /></td></tr> 
<tr> 
<!-- 单 击 “ 下 一 步 ”按钮 ， 用 来 触发 onClick 事件 ， 并 调用 chkanswer() 函 数 --> 
<td><input id = " next2 " name = " next2 " type=" button " dlass=" btn " value =" 下 一 步 "onClick = " return 
chkanswer ( foundanswer ) "></td> 
</tr> 
</form> 
</table> 
</div> 
<!-- 第 3 个 div 标签 ， 样 式 也 为 隐藏 ， 作 用 是 修改 密码 -> 
<div id='third' style="display:none;"> 
<table> 
<form id="modifypwd" name="found" method="post" action="#"> 
<tr><td> 输入 密码 </td></tr> 
<tr><td> 输 入 密码 : </td> 
<td><input id="pwd1" name="pwd1" type="password" class="txt"></td></tr> 
<tr><td> 确 认 密码 : </td> 
<td><input id="pwd2" name="pwd2" ="password" class="txt" /></td> 
</tr> 
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<tr> 
<!-- 单 击 “ 完 成 ”按钮 ， 调 用 chkpwd() 函 数 --> 
<td><input id = " mod " name = “ mod " type = " button " class = " btn " value = "完成 "onClick = " retum 
chkpwd (modifypwd) "></td> 
</tr> 
</form> 


</table> 
</div> 


可 以 看 出 ， 在 上 述 3 个 表单 中 ， 只 有 一 个 表单 默认 情况 下 是 显示 的 ， 其 他 则 为 隐藏 。 只 有 通过 调 
用 不 同 的 js 函数 ， 才 可 以 对 其 他 表单 进行 操作 。 
2. 创建 js 脚本 文件 


found.js 脚本 文件 包含 chkname0、chkanswer0 和 chkpwd03 个 函数 。 其 中 ，chkname() 函 数 的 作用 
是 检查 用 户 输 入 的 会 员 名 称 ， 如 果 存 在 ， 则 使 用 xmlhttp 对 象 去 调用 生成 的 url 进行 处 理 判 断 。 如 果 该 
用 户 存在 ， 则 隐藏 当前 表单 ， 并 显示 下 一 个 表单 ， 最 后 输出 密 保 问 题 。chkname() 函 数 的 代码 如 下 : 
倒 程 16 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\found.js 


function chkname(form){ 

/获取 文本 框 信息 ”/ 

var user = form.user.value; 

让 如 果 为 空 ， 则 输出 提示 */ 

if(user == "X{ 
alert(' 请 输入 用 户 名 '); 
form.user.focus(); 
return false; 

jelse{ 

/否则 ， 生 成 url， 并 调用 xmlhttp 对 象 */ 
var url = "foundpwd.php?user="+user; 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(X{ 
if(xmlhttp.readyState == 4X{ 

var msg = xmlhttp.responseText; 

/如 果 没 有 结果 ， 则 提示 六 

if(msg == '0){ 
alert( 没 有 该 用 户 ， 请 重新 查找 !"); 
form.user.select(); 
retur false; 

}else{ 

/* 否 则 ， 隐 藏 当前 表单 ， 显 示 下 个 表单 ， 并 输出 密 保 问题 */ 
document.getElementByld('first').style.display = 'none'; 
document.getElementByld(second ).style.display = "; 
document.getElementByld(question ).innerHTML = msg; 


} 


} 
xmlhttp.send(null); 
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其 他 两 个 函数 也 使 用 xmlhttprequest 对 象 ， 实 现 方 法 相差 无 几 ， 不 同 之 处 就 是 对 返回 值 的 处 理 ， 
chkanswer() 函 数 隐 藏 当前 表单 ， 显 示 下 一 个 表单 。chkanswerO 函 数 的 代码 如 下 : 
全 性 17 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\found.js 


function chkanswer(form) { 
/* 获 取 已 隐藏 表单 数据 */ 


} 


var user = document.getElementByld(user).value; 
让 获取 密 保 答案 */ 

var answer = form.answer.value; 

让 表单 验证 */ 

if(lanswer == "){ 


jelse{ 
/生成 ur 
var url = "foundpwd.php?user="+user+"&answer="+answer' 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(X{ 
if(xmlhttp.readyState == 4){ 
var msg = xmlhttp.responseText; 
if(msg == "0){ 
alert(' 问 题 回答 错误 '); 
form.answer. select(); 
return false; 
}else{ 
document.getElementByld(second ').style.display = "none'; 
document.getElementByld(third ).style.display = "; 


} 
} 
xmlhttp.send(null); 


而 chkpwd0O 函 数 则 提示 用 户 操作 状态 ， 如 果 成 功 ， 则 关闭 当前 页 。chkpwdO 函 数 的 代码 如 下 : 
倒 程 18 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\found.js 
function chkpwd(form){ 


* 获 取 名 称 和 密码 */ 
var user = document.getElementByld('user ).value; 
var pwd1 = form.pwd1.value; 
var pwd2 = form.pwd2.value; 
仆 省 略 部 分 为 表单 验证 */ 
var url = "foundpwd.php?user="+user+"&password="+pwd1; 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.response Text; 
if(msg == "1'X 
alert(' 密 码 修改 成 功 ， 请 重新 登录 '); 


Window.close(); 
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} 


: 

xmlhttp.send(null); 
7 
3. 创建 数据 处 理 文件 


foundpwd.php 文件 的 功能 是 根据 用 户 输入 的 信息 来 检测 数据 表 中 的 数据 ， 并 根据 不 同 的 输入 信息 返 
回 不 同 的 结果 。 该 文件 的 代码 如 下 : 
倒 程 19 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\foundpwd.php 


<?php 
include_once 'system/system.inc.php'; // 载 入 数据 库 连接 文件 
$name= $_GET['user]; 1/ 获取 表 单 值 


$answer = $_GET[answer]; 
$password = $_GET['password'"; 
Sreback = '0'; 
/如 果 变量 $answer 和 $password 为 空 ， 说 阴 用 户 只 输入 了 会 员 名 称 */ 
if(empty($answer) && empty($password)){ 
/* 生 成 查询 语句 */ 
$namesql = "select * from tb_user where name = ".$name.""; 
/获取 查 询 结果 */ 
$namerst = $conn->execute($namesql); 
/* 如 果 查 询 记录 为 1， 说 明 输入 正确 */ 
if($namerst->recordCount() == 1X{ 
/获取 密 保 问题 ， 并 赋 给 变量 $reback"/ 
$question = $namerst->fields['question ]; 
$reback = $question; 


} 
/如 果 变 量 $answer 不 为 空 ， 说 阴 用 户 输入 了 密 保 答案 */ 
}else if(lempty($answer)){ 
/根据 密 保 答案 和 会 员 名 称 ， 生 成 查询 语句 */ 
$answersql = "select * from tb_user where name = ".$name." and answer = ".$answer.""; 
/返回 查询 结果 ， 并 判断 是 否 有 返回 记录 ， 如 果 有 ， 说 明 输入 正确 %/ 
$answerrst = $conn->execute($answersql); 
if($answerrst->recordCount() == 1X{ 
$reback = "1'; 


} 
/如 果 变 量 $password 不 为 空 ， 说 明 用 户 输入 了 新 密码 */ 
}else if(lempty($password)X{ 
“根据 用 户 名 称 和 新 密码 更 新 记录 */ 
$sql = "select * from tb_user where name = ".$name.""; 
S$arr = array(); 
$rst = $conn->execute($sql); 
if($rst->RecordCount() == 1X 
$arr[password'] = md5($password); 
$updateSQL = $conn->GetUpdateSQL($rst$arrtrue); 
$conn->execute($updateSQL); 
Sreback = "1'; 
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} 
} 
让 输出 返回 值 */ 
echo $reback; 
?> 
4. 加 载 模板 页 


因为 所 有 登录 模块 的 模板 都 不 需要 或 者 只 需要 传递 一 两 个 变量 ,所 以 PHP 加 载 页 的 内 容 比较 简单 。 
找 回 密码 页 面 的 代码 如 下 : 
倒 性 20 ”代码 位 置 : 光盘 \TM\10\ShoppingSystem\found.php 


<?php 
include_once 'system/system.inc.php'; // 载 入 配置 文件 
$smarty->assign('title',' 找 回 密码 '); // 向 模板 中 传递 title 变量 
$smarty->display(found.html"); /显示 found.html 模板 页 
?> 


10.8 会员 信息 模块 设计 


10.8.1 会 员 信 息 模块 概述 


用 户 登录 后 ， 即 可 看 到 会 员 信息 模块 。 在 这 里 ， 可 以 进 
行 查看 或 修改 个 人 信息 及 密码 、 查 看 购物 车 和 安全 退出 等 操 
作 。 本 节 只 对 会 员 信息 模块 中 的 “会 员 中 心 ” 和 “安全 退出 ” 
进行 讲解 ， 关 于 “查看 购物 车 ”将 在 商品 模块 中 进行 介绍 。 
会 员 信息 模块 的 运行 结果 如 图 10.20 所 示 。 


。 
10.8.2 会 员 信息 模块 技术 分 析 ES 


在 会 员 信息 模块 中 , 以 session 变量 中 存储 的 用 户 名 称 为 
条 件 ， 从 会 员 信息 表 中 查询 出 会 员 信息 ， 并 且 将 会 员 信息 存 
储 到 模板 变量 中 , 最 后 在 模板 页 中 输出 会 员 信息 。memberphp 
文件 的 代码 如 下 : 图 10.20 会 员 信息 模块 的 运行 结果 

倒 竹 21 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\member.php 


<?php 
session_start(); /初始化 session 变量 
include_once 'system/system_.inc.php'; // 包 含 配置 文件 
/查找 用 户 资料 纪 
$sql = "select * from tb_user where name = ".$_SESSION[member].""; // 执 行 查询 操作 


Srst = $conn->execute($sql); 
$arr = $rst>GetArray(); 
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$smarty->assign(arr,$arrf0]); /将 查询 结果 赋 给 模板 变量 


?> 


memberphp 文件 中 查询 出 的 数据 是 会 员 信 息 模板 功能 实现 的 根本 。 
10.8.3 会 员 中 心 


国 用户 注册 使 用 的 数据 表 : tb_user 
当 单 击 “ 会 员 中 心 ” 超 链 接 时 ， 会 回 传 给 当前 页 一 个 page 值 ， 当 前 页 根据 这 个 page 值 来 载 入 
memberphp 文件 。 


1. 创建 PHP 页 面 


与 登录 模块 设计 不 同 ， 本 节 首 先 来 创建 PHP 页 面 。 因 为 该 模块 中 的 模板 页 需要 使 用 数据 库 中 的 数 
据 及 一 些 动态 信息 ， 这 些 都 需要 在 PHP 页 中 先行 获取 及 处 理 ， 然 后 再 传 给 模板 页 。 会 员 中 心 页 面 的 代 
码 请 参考 技术 分 析 中 的 内 容 。 
2. 创建 模板 页 
该 模块 包括 查看 信息 模板 及 修改 密码 模板 。 首 先 介绍 查看 信息 模板 ， 代 码 如 下 : 
倒 程 22 代码 位 置 ， 光盘 \TM\10\ShoppingSystem\membershow.html 
<!-- 显 示 超 链接 --> 
<p>{$smarty.session.member}<a href = ' ?page=hyzx ' > 查看 信息 </a><a href = ' ?page = hyzx&action=modify' 
id="mem"> 修 改 密码 </a></p> 
<table> 
<form id = "member " name = "member " method = " post " action = " modify_info_chk.php " onsubmit = " return 
mem( member ) "> 
<!-- 显 示 当 前 会 员 信息 ， 此 处 为 不 可 更 改 信息 ”一 > 
<tr><td colspan="2">{$arrname} 信 息 〈 不 可 更 改 信息 ) </td></tr> 
<!-- 因 为 格式 相同 ， 所 以 只 显示 一 条 ， 其 他 省 略 --> 
<tr><td> 会 员 编 号 : </td><td >&nbsp;{$arr.id}</td></tr> 


<tr><td>{$arr.name} 信 息 (可 更 改 信息 ) </td></tr> 
<!-- 可 以 修改 的 信息 ， 放 到 表单 元 素 中 --> 
<tr><td> 真 实 姓 名 : </td> 
<td><input id="realname" name="realname" type="text" value="{$arr.realname}" /></td></tr> 
<tr><td> 身 份 证 号 : </td> 
<td><input id="card" name="card" type="text" value="{$arr.card}" /></td></tr> 
<tr><td> 移 动 电话 : </td> 
<td><input id="tel" name="tel" type="text" value="{$arr.tel}"></td></tr> 
<tr><td> 固 定 电话 : </td> 
<td><input id="phone" name="phone" type="text" value="{$arr.phone}" /></td></tr> 
<tr><td>Email: </td> 
<td><input id="email" name="email" type="text" value="{$arr.Email}" /></td></tr> 
<tr><td>QQ 号 : </td> 
<td><input id="qq" name="qq" type="text" value="{$arr.QQ}Y" /></td></tr> 
<tr> 
<td> 邮 编 : </td> 
<td><input id="code" name="code" type="text" value="{$arr.code}" /></td></tr> 


_ 国 
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<tr><td> 地 址 : </td> 
<td><input id="address" name="address" type="text" value="{$arr.address}" /></td></tr> 
<tr> 
<td><input name="enter" type="submit" id="enter value=" 修 改 " /> 
<input name="reset" type="reset" id="reset" value=" 重 置 " /></td></tr> 
</form> 
</table> 


修改 密码 模板 比 信息 模板 要 简单 一 些 ， 因 为 不 需要 额外 的 传 值 。 模 板 代 码 如 下 : 
全 性 23 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\modifypwd.html 


<!-- 载 入 脚本 文件 -> 
<script language="javascript" src="js/member.js"></script> 
<p>{$smarty.session.member}<a href='?page=hyzx'> 查 看 信息 </a><a href='?page=hyzx&action=modify' > 修改 
密码 </a></p> 
<table> 
<! 一 修改 密码 表单 一 > 
<form id =" member " name =" member " method = " post " action = " modify_pwd_chk.php " onsubmit = " return 
pwd (member) "> 
<tr><td > 修改 密码 </td></tr> 
<tr><td > 原 密码 : </td><td><input id="old" name="old" type="password" /></td></tr> 
<tr><td> 新 密码 : </td><td><input id="new1" name="new1" type="password" /></td></tr> 
<tr><td> 确 认 密 码 : </td><td><input id="new2" name="new2" type="password" /></td></tr> 
<tr><td><input id="enter" name="enter" type="submit" value=" 修 改 " /></td></tr> 
</form> 
</table> 


3. 创建 脚本 文件 

该 模块 的 脚本 文件 和 用 户 注册 模块 类 似 ， 都 是 对 信息 的 合法 性 进行 验证 ， 如 信息 是 否 为 空 、 是 否 
符合 规范 等 ， 这 里 不 再 资 述 。 

4. 创建 处 理 页 


当 信息 验证 通过 后 ， 系 统 将 跳 转 到 处 理 页 进行 信息 处 理 。 本 模块 处 理 页 分 信息 修改 和 密码 修改 两 
个 页 面 。 首 先 介 绍 信息 修改 页 ， 代 码 如 下 : 
倒 程 24 ”代码 位 置 ， 光盘 \TMN10\ShoppingSystemvmodify_info_chk.php 


<?php 
session_start(); /开启 session 支持 
include_once ‘system/system.inc.php’; // 载 入 数据 库 连 接 文件 
$sql = 'select * from tb_user where id = '.$_SESSION['id]; /根据 id 生成 SQL 查询 语句 
S$rst = $conn->execute($sql); /| 执行 SQL 语句 ， 返 回 查询 结果 集 
$mod = array(); /声明 数组 
/* 将 修改 信息 添加 到 数组 中 */ 


$mod['realname'] = $_POST[realname']; 
4$mod['card] = $_POST[card"]; 
$mod['tel] = $_POST[tel]; 
4$mod[phone] = $_POST[phone'; 
$mod[Email] =$_POST[email]; 
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4$mod[QQ] = $_POST['qq]; 
$mod['code] = $_POST[code']; 
S$mod[address] = $ i 
J 
$updateSQL = 5 $mod); /生成 更 新 语句 
if($conn->execute($updateSQL)) 
echo "<script>alert(' 修 改 成 功 ');location=('index.php');</script>"; // 修 改 成 功 ， 跳 到 首页 
else 
echo "<script>alert(' 修 改 失 败 ');history.go(-1);</script>"; // 和 否则 ， 返 回 上 一 页 
?> 


密码 修改 页 的 操作 流程 也 十 分 类 似 ， 只 是 更 新 的 数组 要 小 得 多 ， 只 有 一 个 字段 。 密 码 修改 页 的 代 
码 如 下 : 
倒 程 25 代码 位 置 ， 光 盘 \TM\10\ShoppingSystem\modify_pwd_chk.php 


<?php 
session_start(); /开启 session 支持 
include_once 'systerysystem.inc.php';; // 载 入 数据 库 连接 文件 
$oldpwd = md5($_POST[old]); // 对 密码 进行 md5 加 密 


/根据 id 和 md5 加 密 后 的 密码 查询 用 户 */ 
$sql = 'select * from tb_user where id = '.$_SESSION['id].' and password = \".$oldpwd.\"’; 


Srst = $conn->execute($sql); // 返 回 查询 结果 
$mod = array(); // 创 建 更 新 数组 
$mod['password] = md5($_POST[new1]); /为 新 密码 进行 md5 加 密 
$updateSQL = $conn->GetUpdateSQL($rst$mod); /生成 更 新 语句 


if($conn->execute($updateSQL)) 

echo "<script>alert(' 修 改 成 功 ');location=('index.php');</script>"; // 修 改 成 功 ， 返 回首 页 
else 

echo "<script>alert(' 修 改 失败 ');history.go(-1);</script>"; // 否 则 ， 返 回 上 一 页 


10.8.4 ”安全 退出 


当 用 户 需 要 离开 网 站 时 ， 可 以 单 击 “ 安 全 退出 ” 超 链接 来 调用 logout0 函 数 ， 当 用 户 确认 退出 后 ， 
则 跳 转 到 logout 页 面 ， 销 毁 session 并 回 到 首页 。 安 全 退出 所 涉及 的 页 面 及 代码 如 下 : 
倒 程 26 ”代码 位 置 ， 光 盘 \TM\10\ShoppingSystem\js\info.js 


function logout(){ 


说 confirm(" 确 定 要 退出 登录 吗 ? “")){ /输出 选择 框 ， 用 户 可 以 单 击 “确认 ”或 “取消 ”按钮 
window.open(logout.php'”parent',",false); 。“”// 如 果 用 户 确认 退出 ， 则 打开 logout.php 页 
jelse 
return false; 
} 
倒 避 27 代码 位 置 ， 光盘 \TM\10\ShoppingSystem\logout.php 
<?php 
session_start(); /开启 session 支持 
session_destroy(); // 销 毁 session 
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echo '<script>alert(' 用 户 已 安全 退出 \);location=(\'index.php\);</script>"; // 回 到 首页 
i 


10.9 商品 显示 模块 


10.9.1 商品 显示 模块 概述 


本 系统 为 用 户 提供 了 不 同 的 商品 展示 方式 ， 包 括 推 
荐 商品 、 最 新 商品 、 热 门 商品 等 ， 能 够 使 消费 者 有 目的 
地 选 购 商 品 。 每 个 展示 方式 中 包括 商品 的 详细 信息 显 
示 ， 为 用 户 购买 商品 提供 可 靠 的 依据 。 本 系统 商品 显示 - pr 
模块 的 运行 结果 如 图 10.21 所 示 。 rr i bi 

因为 推荐 商品 、 最 新 商品 和 热门 商品 的 实现 方法 和 s 
过 程 基本 相同 ， 所 以 本 节 只 讲解 推荐 商品 模块 。 其 他 功 
能 相关 代码 可 参见 光盘 中 的 源 程序 。 


10.9.2 ”商品 显示 模块 技术 分 析 


二 ed 机 Dod 


商品 显示 功能 实现 的 关键 就 是 如 何 从 数据 库 中 读 rr 
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取 商 品 信息 ， 如 何 完成 数据 的 分 页 显示 。 在 定义 SQL 
语句 时 ， 首 先 判断 字段 isnom 的 值 ， 如 果 该 字段 为 1， 图 10.21 商品 显示 模块 页 面 
即 为 推荐 ， 和 否则 为 不 推荐 ， 然 后 再 定义 数据 降 守 排列 ， 
并 设置 每 页 显示 4 条 记录 ， 这 就 是 完成 商品 显示 的 查询 语句 。 代 码 如 下 : 

倒 程 28 ”代码 位 置 : 光盘 \TM\10\ShoppingSystem\nominate.php 


<?php 
include_once 'system/system.inc.php'; /连接 数据 库 
$sql = "select id,name,pics,m_price,v_price from tb_commo where isnom = 1 order by id desc limit 4"; 
$nomarr = $admindb->ExecSQL($sql,$conn); /| 执行 查询 语句 
$smarty->assign(nomarr,$nomarrj; /将 查询 到 的 数据 赋 给 模板 变量 

?> 


最 后 , 定义 模板 文件 , 通过 section 语句 循环 输出 存储 在 模板 变量 中 的 数据 ， 即 完成 商品 展示 的 操作 。 
section 是 Smarty 模板 中 的 一 个 循环 语句 ， 该 语句 用 于 复杂 数组 的 输出 。 其 语法 如 下 : 


{section name="sec_name" loop=$arr_name start=num step=num} 
参数 含义 : name 是 该 循环 的 名 称 ; loop 为 循环 的 数组 ，start 表示 循环 的 初始 位 置 ， 如 start=2， 那 


么 说 明 循环 是 从 loop 数组 的 第 2 个 元 素 开始 的 ，step 表示 步 长 ， 如 step=2， 那 么 循环 一 次 后 ， 数 组 的 
指针 将 向 下 移动 两 位 ， 以 此 类 推 。 
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10.9.3 ”商品 显示 模块 的 实现 过 程 


国 商品 显示 使 用 的 数据 表 : tb_ commo、tb_user 
在 10.9.2 节 中 已 经 对 商品 显示 所 使 用 的 技术 、 方 法 进行 了 概述 ， 下 面 就 介绍 一 下 它 具 体 的 过 程 。 
(1) 创建 nominate.php 文件 ， 从 数据 库 中 读 取出 推荐 商品 的 数据 ， 并 将 数据 存储 到 模板 变量 中 ， 
其 代码 可 以 参考 10.9.2 节 。 
(2) 创建 nominate.html 模板 页 ， 应 用 section 语句 输出 商品 信息 ， 并 添加 相应 的 操作 按钮 或 链接 。 
模板 页 中 一 共有 3 个 事件 ， 即 显示 更 多 商品 、 查 看 商品 和 放 入 购物 车 。 
回 ” 当 单 击 “ 更 多 商品 ” 超 链接 时 ， 将 会 重新 加 载 本 页 面 ， 并 传递 一 个 page 变量 。switch 语句 会 
根据 page 值 来 显示 。 
当 单 击 “ 查 看 详情 ”按钮 时 ， 将 触发 onClick 事件， 并 将 调用 openshowcommo0 函 数 ， 同 时 ， 
商品 id 会 作为 函数 的 唯一 参数 被 传递 进去 。 
当 单 击 “ 购 买 ”按钮 时 ， 同 样 会 触发 onClick 事件 ， 并 调用 buycommo() 函 数 ， 唯 一 的 参数 也 
是 商品 的 id。 
商品 模板 页 面 的 代码 如 下 : 
倒 程 29 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\nominate.html 


<!-- 载 入 css 样式 -> 
<link href="css/nominate.css" rel="stylesheet" type="text/css" /> 
<link href="css/links.css" rel="stylesheet" type="text/css" /> 
<!-- 载 入 js 脚本 文件 --> 
<script language="javascript" src="js/createxmlhttp.js"></script> 
<script language="javascript" src="js/showcommo.js"></script> 
<table width="643" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td colspan="6" background="images/default_14.gif" width="636" height="39" align="right" valign= 
"middle"><a href="?page=nom" class="|k">&gt;&gt;more&it;&lt;</a></td> 
</tr> 
<tr> 
{section name=id loop=$nomarr} 
<td width="145" height="185" align="left" valign="top"> 
<table width="145" border="0" cellpadding="0" cellspacing="0" > 
<tr> 
<td height="100" align="center" valign="middle"><img src="{$nomarrlid].pics}” width="100" height="80" 
alt="{$nomarr[idj.namej}" style="border: 1px solid #f0f0f0;" ></td> 
</tr> 
<tr> 
<td height="32" align="center" valign="middle"><input id="showinfo" name="showinfo" type="button" 
value="" class="showinfo" onclick="openshowcommo({$nomarr[id].id})"/>&nbsp;<input id="buy"” name="buy" 
type="button" value="" class="buy" onclick="returm buycommo({$nomarr[lid].id})" /></td> 
</tr> 
</table> 
</td> 
{/section} 
</tr> 
</table> 
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(3) 创建 showcommo.js 脚本 文件 。 当 单 击 “ 查 看 商品 ”按钮 时 ， 系 统 会 弹出 一 个 新 的 页 面 ， 并 
显示 商品 的 详细 信息 ; 当 单 击 “ 购 买 ” 按 钮 时 ， 该 商品 将 会 被 放 到 当前 用 户 的 购物 车 中 ， 如 果 没有 登 
录用 户 或 商品 已 添加 ， 则 会 提示 错误 信息 。js 脚本 文件 的 代码 如 下 : 
倒 程 30 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\showcommo.js 
/* 查 看 商品 信息 函数 ， 将 打开 一 个 新 页 面 */ 
function openshowcommo(key){ 
open(showcommo.php?id='+key”blank', width=560 height=300',false); 


} 
/将 购买 商品 添加 到 购物 车 中 ， 将 在 10.10 节 中 讲解 */ 
function buycommo(keyi{f 


} 


10.10 ”购物 车 模块 设计 
10.10.1 ”购物 车 模块 概述 


购物 车 在 电子 商务 平台 中 是 前 台 客户 端 程序 中 非常 关键 的 一 个 功能 模块 。 购 物 车 的 主要 功能 是 保 
留用 户 选择 的 商品 信息 ， 用 户 可 以 在 购物 车 内 设置 选 购 商品 的 数量 ， 显 示 选 购 商 品 的 总 金额 ， 还 可 以 
清除 选择 的 全 部 商品 信息 ,重新 选择 商品 信息 。 购 
物 车 页 面 的 运行 结果 如 图 10.22 所 示 。 

购物 车 模块 主要 实现 添加 商品 、 删 除 商 品 和 更 : 

改 数量 等 操作 。 ER Rtn | 


10.10.2 ”购物 车 模块 技术 分 析 图 1022 购物 车 而 


购物 车 功能 实现 最 关键 的 部 分 就 是 如 何 将 商品 添加 到 购物 车 ， 如 果 不 能 完成 商品 的 添加 ， 那 么 购 
物 车 中 的 其 他 操作 都 没有 任何 意义 。 
在 商品 显示 模块 中 ， 单 击 商品 中 的 “购买 ”按钮 ， 将 商品 放 到 购物 车 中 ， 并 进入 到 “购物 车 ”页 
面 。 单 击 “ 购 买 ” 按 钮 调用 buycommo0 函 数 ， 购 买 商品 的 id 是 该 函数 的 唯一 参数 ， 在 buycommo() 函 
数 中 通过 xmlhttp 对 象 调用 chklogin.php 文件 ， 并 根据 回 传 值 作出 相应 处 理 。buycommo() 函 数 的 代码 
如 下 : 
倒 程 31 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\showcommo.js 
六 
* 添 加 商品 ， 同 时 检查 用 户 是 否 登录 ， 商 品 是 否 重复 等 
刀 
function buycommo(keyX{ 
/* 根 据 商 品 id， 生 成 url*/ 
var url = "chklogin.php?key="+key; 
/使 用 xmlhttp 对 象 调用 chklogin.php 页 */ 
xmlhttp.open("GET",url,true); 
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xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
让 用 户 没 有 登录 */ 
if(msg == "2){ 
alert(' 请 您 先 登录 ); 
retur false; 
jelse if(msg == '3){ 
让 商品 已 添加 */ 
alert( 该 商品 已 添加 '); 
retum false; 
Jelse{ 
/显示 购物 车 */ 
location='index.php?page=shopcar'; 
} 
} 


} 
xmlhttp.send(null); 


在 chklogin.php 文件 中 将 商品 添加 到 购物 车 中 。chklogin.php 页 的 代码 如 下 : 
倒 程 32 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\chklogin.php 


<?php 
session_start(); /开启 session 支持 
Ar 
*1 表示 添加 成 功 
*2 表示 用 户 没有 登录 
*3 表示 商品 已 添加 过 
*4 表示 添加 时 出 现 错误 
*5 表示 没有 商品 添加 
学 
include_once 'system/system.inc.php'; // 载 入 数据 库 连接 文件 
Sreback = '0'; /声明 返回 值 
if(empty($_SESSION['member])X // 如 果 session 值 为 空 ， 说 明 没有 登录 
Sreback = '2'; /参见 头 注释 
Jelse{ 
$key = $_GET[key']; /获取 商品 id 
if(Skey == " 
S$reback = '5'; 
}else{ 
$id = (nt)$_SESSION[id]， /获取 登录 会 员 id 号 
$boo = false; /设置 一 个 布尔 变量 ， 初 始 为 false 
$addshop = array(); /数据库 更 新 数组 
$sql = "select id,shopping from tb_user where id = ".$id; // 生 成 当前 用 户 的 查询 语句 
S$rst = $conn->execute($sql); /1 返回 当前 用 户 记录 集 
$shopcont = S$rst->fields[ shopping"]; /找到 shopping 字段 
if(lempty($shopcont)X{ // 如 果 字 段 不 为 空 ， 说 明 已 有 商品 
$arr = explode(@',$shopcont); /将 字段 用 “@” 拆 分 并 另存 数组 
foreach($arr as $valueX{ // 循 环 输出 数组 内 容 
if($key == $value[0]){ // 判 断 需 要 添加 的 商品 是 否 存 在 
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Sreback ='3"; /参见 头 注释 
$boo = true; /同时 布尔 变量 变 为 true 
break; /停止 循环 
} 
} 
if(I$boo){ /如 果 布 尔 值 没有 改变 ， 再 添加 商品 


/将 商品 数量 默认 设 为 1， 并 连同 商品 id 保存 到 字 串 中 */ 
$shopcont .= '@'.$key ,1 
让 将 商品 字 串 保存 到 更 新 数组 中 */ 
$addshop['shopping'] = $shopcont; 
让 生成 更 新 语句 */ 
SupdateSQL = $conn->GetUpdateSQL(S$rst,$addshop); 
更 新 数据 表 */ 
if(false == $conn->execute($updateSQL)}{ 
Sreback = "4'; 
Jelse{ 
Sreback = "1'; 
} 


} 
}else{ 
/如 果 shopping 字段 为 空 ， 则 直接 更 新 */ 
$tmparr = $key",1"; 
$addshop['shopping] = $tmparr; 
$updateSQL = $conn->GetUpdateSQL(S$rst,$addshop); 
if(false == $conn->execute($updateSQL)X 


Sreback = "4'; 
jelsef 
Sreback = "1'; 
; 
} 
} 
} 
echo $reback; 


?> 


通过 分 析 上 述 代码 可 知 ，shopping 字段 保存 的 是 购物 车 中 的 商品 信息 ， 一 条 商品 信息 包括 两 部 分 ， 
即 商品 id 和 商品 数量 ， 其 中 商品 数量 默认 为 1。 两 部 分 之 间 使 用 逗号 “,” 分 阳 ， 如 果 添 加 多 个 商品 ， 
则 每 个 商品 之 间 使 用 “@” 分 隔 。 

成 功 完成 商品 的 添加 操作 后 ， 即 可 进入 到 购物 车 页 面 ， 执 行 其 他 的 操作 。 


10.10.3 ”购物 车 展示 


国 购物 车 展示 使 用 的 数据 表 : tb_ commo、tb_user 

购物 车 页 面 分 PHP 代码 页 和 Smarty 模板 页 。 在 PHP 代码 页 中 ,首先 读 取 tb_user 数据 表 中 shopping 
字段 的 内 容 ， 如 果 字 段 为 室 ， 则 输出 “购物 车 中 暂时 没有 商品 !”: 如 果 数 据 库 中 有 数据 ， 则 循环 输出 
数据 ， 并 将 商品 信息 保存 到 数组 中 ， 再 传 给 模板 页 。 购 物 车 页 面 的 代码 如 下 : 
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倒 程 33 ”代码 位 置 : 光盘 \TM\10\ShoppingSystem\myshopcar.php 
<?php 


/* 载 入 数据 库 连 接 文 件 和 Smarty 配置 文件 */ 
include_once 'system/system.inc.php'; 
/根据 登录 会 员 的 用 户 名 生成 查询 语 名 六 
$sql1 = "select id,shopping from tb_user where name =".$_SESSION[member].”"; 
让 执行 查询 语句 返回 结果 集 */ 
Srst = $conn->execute($sql1); 
/如 果 字 段 为 空 ， 直 接 输出 “购物 车 中 暂时 没有 商品 信息 !”， 并 结束 运行 */ 
if$rst->fields['shopping] == " 
echo "<p>"; 
echo ' 购 物 车 中 暂时 没有 商品 必 
exit(); 


} 
广 如 果 有 结果 集 ， 则 将 结果 集 另存 为 数组 */ 
S$tmparr = $rst->GetAssoc(); 
$commarr = array(); 
/* 输 出 shopping 字段 值 */ 
foreach($tmparras S$value){ 
/* 使 用 “@” 进 行 分 隔 ， 得 到 数组 */ 
S$tmpnum = explode(@',$value); 
"得 到 购买 商品 种 类 */ 
$shopnum = count($tmpnum); 
/商品 总 价值 */ 
$sum = 0; 
/* 数 组 循环 */ 
foreach($tmpnum as $key => $vI){ 
/使 用 逗号 “,” 拆 分 单一 的 商品 信息 
$s_commo = explode(,,$vl); 
/根据 保存 的 商品 信息 查找 商品 %/ 
$sql2 = "select id,name,m_price,fold,v_price from tb_commo"; 
$commsql = $sql2." where id = ".$s_commof0]; 
返回 结果 集 */ 
$commrst = $conn->execute($commsql); 
让 将 结果 集 另存 数组 */ 
$arr = $commrst->GetArray(); 
让 将 需要 显示 的 信息 处 理 后 保存 到 数组 中 */ 
$arr[0l[num'] = $s_commo[1]; 
$arr[0][total] = $s_commo[1]*$arr[0][v_price]; 
4$sum += $arr[O]['total]; 
$commarrf$key] = $arr[0]; 


} 

让 将 得 到 的 两 个 数组 和 总 消费 金额 传递 给 模板 */ 
$smarty->assign('shoparr,$shopnum); 
4$smarty->assign(commarr,$commarrj; 
$smarty->assign('sum',$sum); 


$smarty->assign('title', 我 的 购物 车 "); 


?> 
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商品 的 模板 页 不 仅 要 负责 用 户 购 买 商品 信息 的 输出 ， 而 且 还 要 提供 可 以 对 商品 进行 修改 、 删 除 等 
操作 的 事件 接口 。 模 板 页 代码 如 下 : 
倒 性 34 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\myshopcar.html 


<!-- 载 入 使 用 到 的 js 脚本 文件 --> 
<script language="javascript" src="js/createxmlhttp.js"></script> 
<script language="javascript" src="js/shopcarjs"> </script> 
<table> 
<! 一 购物 车 表单 --> 
<form id="myshopcar" name="myshopcar" method="post" action="#"> 
<tr><td> 我 的 购物 车 </td></tr> 
<tr><td >&nbsp;</td><td> 商 品名 称 </td><td> 购 买 数量 </td><td> 市 场 价格 </td> 
<td> 会 员 价 格 </td><td> 折 扣 率 </td><td> 合 计 </td></tr> 
<!-- 使 用 foreach 表单 循环 输出 商品 种 类 --> 
{foreach key=key item=item from=$commarr} 
<!-- 删 除 复 选 框 ， 名 称 用 数组 表示 ， 每 个 复 选 框 的 值 为 商品 的 id 号 --> 
<tr><td><input id="chk" name="chk[]" type="checkbox" value="{$item.idj"></td> 
<!-- 显 示 该 商品 的 名 称 --> 
<td><div id = "c_name{$key}"> &nbsp;{S$item.name}</div></td> 


<td> 
<!-- 商 品 数量 文本 框 ， 每 个 复 选 框 的 名 称 为 "cnum"+key 值 --> 
<input id = " cnum{$key} " name = " cnum{$key} " type = " text " value = " {S$item.num} " onkeyup = " cvp 
( {$key},{$item.v_price},{$shoparr})"></td> 
<!-- 商 品 的 其 他 信息 --> 
<td><div id="m_price{$key}">&nbsp;{$item.m_price}</div></td> 
<td><div id="v_price{$key}">&nbsp;{S$item.v_price}</div></td> 
<td><div id="fold{$key}">&nbsp;{$item.fold}</div></td> 
<td><div id="total{$key}">&nbsp;{$item.total}</div></td> 
</tr> 
<!-- 循 环 结束 -> 
{/foreach} 
<tr> 
<td colspan="3"> 
<!-- 全 选 、 反 选 和 删除 操作 -> 
<a href="#" onclick="retum alldeltmyshopcar)"> 全 选 </a> 
<a href="#" onclick="return overdel(myshopcar);"> 反 选 </a> 
<input type="button" value=" 删 除 选择 " onClick = 'return del(myshopcar);'></td> 


<td> 
<!-- 继 续 购物 一 > 
<input type="button" value=" 继 续 购物 " onclick="return conshop(myshopcar)" /></td> 
<td> 
<!-- 将 当前 用 户 名 保存 到 隐藏 域 中 --> 
<input id="uid" name="uid" type="hidden" value="{$smarty.session.member}" > 
<!- 结 算 ， 去 收银 台 --> 
<input type="button" class="btn" value=" 去 收银 台 " onclick="return formset(form)" /></td> 
<td colspan="2"><div id='sum'> 共 计 : {$sum}&nbsp; 元 </div></td> 
</tr> 
</form> 
</table> 
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10.10.4 ”更 改 商品 数量 


国 ” 更改 商 品 数量 使 用 的 数据 表 : tb_ commo、tb_user 
对 于 新 添加 的 商品 ,默认 的 购买 数量 为 1,， 在 购物 车 页 面 可 以 对 商品 的 数量 进行 修改 。 当 商品 数量 
发 生变 化 时 商品 的 合计 金额 和 商品 总 金额 会 自动 发 生 改变 ， 该 功能 是 通过 触发 text 文本 域 的 onkeyup 
事件 调用 cvp0) 函 数 实现 。cvp0 函 数 有 3 个 参数 ， 分 别 是 商品 id、 商 品 单价 和 商品 类 别 。 
首先 ， 通 过 商品 的 id 可 以 得 到 要 修改 商品 的 相关 表单 和 标签 属性 。 然 后 ， 通 过 商品 单价 和 输入 的 
商品 数量 计算 该 商品 的 合计 金额 。 接 着 ， 使 用 for 循环 得 到 其 他 商品 的 合计 金额 。 最 后 ， 将 所 有 的 合计 
金额 累加 ， 并 输出 到 购物 车 页 面 。cvpO 函 数 的 代码 如 下 : 
倒 程 35 代码 位 置 ， 光盘 \TMNV10\ShoppingSystemNjs\vshopcarjs 
/自动 刷新 总 金额 
*key: 商品 id 
* vpr: 商品 单价 
* shoparr: 商品 种 类 数 
于 
function cvp(key, vpr, shoparm)\{ 


var n_pre = ‘total'; lldiv 标签 的 id 前缀 
varnum = 'cnum'+keytoString(); /根据 key 值 生成 文本 域 的 id 值 
vartotal = n_pre+keytoString(); /根据 key 值 生成 div 标签 的 id 值 
vart_number = document.getElementByld(num).value; ”// 获 取 输 入 的 商品 数量 
var tt| = t_number * vpr; // 根 据 商 品 数量 和 单价 ， 计 算 商 品 金 额 
document.getElementByld(total).innerHTML = ttl; // 更 新 商品 总 金额 
varsm = 0; 1// 所 有 商品 的 总 金额 ， 初 始 为 0 
for(var i = 0; i < shoparr i++){ // 根 据 商品 种 类 数 ， 循 环 得 到 其 他 商品 金额 
var aaa = document.getElementByld(n_pre+i.toString()).innerText' 
sm += parselnt(aaa); // 将 所 有 商品 金额 累加 
} 


document.getElementByld('sum').innerHTML = ' 共 计 : '+sm+' 元 '; 


这 里 所 更 改 的 商品 数量 ， 并 没有 被 保存 到 数据 库 中 ， 如 果 希 望 保 存 ， 那 么 单 击 “ 继 续 购物 ”按钮 ， 
则 可 以 将 商品 数量 更 新 到 数据 库 中 。 


10.10.5 删除 商品 


国 删除 商品 使 用 的 数据 表 : tb_commo、tb_user 

当 对 添加 的 商品 不 满意 时 ， 可 以 对 商品 进行 删除 操作 。 操 作 流程 为 : 首先 选中 要 删除 的 商品 前 面 
的 复 选 框 ， 如 果 全 部 删除 ， 则 可 以 单 击 “ 全 选 ”或 “反选 ”按钮 ;然后 单 击 “ 删 除 选择 ”按钮 ， 在 弹 
出 的 警告 框 中 单 击 “ 确 定 ”按钮 ， 商 品 将 被 全 部 删除 。 删 除 商品 的 页 面 结果 如 图 10.23 所 示 。 

所 有 的 删除 操作 都 是 通过 js 脚本 文件 shopcarjs 来 实现 ， 相 关 的 函数 包括 alldel() 函 数 、overdel0) 函 
数 和 del() 函 数 。 
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寺 东 肝 院 调 品 冬运 枉 


时 十， 各 面 


10.23 ”删除 商品 的 流程 


alldel0 函 数 和 overdelO 函 数 实现 的 原理 比较 简单 ,通过 触发 onClick 事件 来 改变 复 选 框 的 选中 状态 。 
代码 如 下 : 
倒 程 36 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\shopcarjs 


/全 部 选择 
function alldel(form){ 
var leng = form.chk.length; // 获 取 复 选 框 数量 
if(leng==undefined){ // 如 果 等 于 undefined， 说 阴 只 有 一 个 复 选 框 
if(!form.chk.checked) 
form.chk.checked=true; // 将 复 选 框 置 于 选中 状态 
}else{ /和 否则， 说明 有 多 个 复 选 杠 
for( vari= 0; i< leng; i++) // 使 用 for 循环 ， 将 所 有 复 选 框 选中 
if(!form.chk[i].checked) 
form.chk[il.checked = true; 
} 
return false; 


} 
/* 反 选 ， 就 是 将 选中 的 复 选 框 取消 选中 ， 而 未 被 选中 的 复 选 框 则 被 选中 */ 


function overdel(formX{ 
var leng = form.chk.length; // 获 取 复 选 框 个 数 
if(leng==undefined}{ // 如 果 leng 为 undefiend， 说 明 只 有 一 个 复 选 框 
if(!form.chk.checked) 
form.chk.checked=true; 
else 
form.chk.checked=false; 
Jelse{ // 否 则 ， 说 明 有 多 个 复 选 框 
for( vari= 0;i< leng; i++) /使 用 for 循环 ， 将 所 有 复 选 框 操作 
4 
if(!form.chkli].checked) /根据 复 选 框 checked 属性 的 情况 ， 进 行 反 向 选择 


form.chk[li].checked = true; 
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else 
form.chk[il.checked = false; 
} 


retur false; 


} 


使 用 alldel0 或 overdel0 选 中 复 选 框 后 ， 即 可 调用 del0 函 数 来 实现 删除 功能 。del0 函 数 首 先 使 用 for 
循环 ,将 被 选中 复 选 框 的 value 值 取出 并 存 成 数组 ， 然 后 根据 数组 生成 url， 并 使 用 xmlhttp 对 象 调用 这 


个 url， 当 处 理 完毕 后 ， 根 据 返 回 值 弹出 提示 或 刷新 本 页 。 该 函数 的 代码 如 下 : 


倒 程 37 代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\shopcarjs 

让 删除 记录 */ 

function del(formX{ 
if(Iwindow.confirm(' 是 否 要 删除 数据 ??')) 


jelse{ 
var leng = form.chk.length; 
if(lleng==undefinedX{ 
if(!form.chk.checked){ 
alert(' 请 选取 要 删除 数据 !"); 
Jelse{ 
rd = form.chk.value; 
var url = "delshop.php?rd='+rd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = delnow; 


xmlhttp.send(null); 


} 
}else{ 
var rd=new Array(); 
varj=0; 
for( vari = 0; i < leng; i++) 
{ 
if(form.chk[i].checkedX{ 
rd[j++] = form.chk[i].value; 
站 
} 
mss™ 
alert(' 请 选取 要 删除 数据 !"); 
Jelse{ 
var url = "delshop.php?rd="+rd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = delnow; 


xmlhttp.send(null); 


} 
} 


return false; 


3} 
人 显示 状态 */ 


// 返 回复 选 框 的 长 度 


/如果 只 有 一 个 复 选 框 ， 且 处 于 选中 状态 
/将 复 选 框 的 value 值 直接 赋 给 变量 rd 
/根据 rd 生成 url 

// 调 用 xmlhttp 对 象 

// 调 用 delnow() 函 数 


// 如 果 复 选 框 为 多 个 
/声明 数组 
// 循 环 检查 复 选 框 状态 


// 将 被 选中 复 选 框 的 value 值 存 到 rd 内 


/根据 rd 生成 ul 


@ 
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function delnow(){ 
if(xmlhttp.readyState == 4){ 
if(xmlhttp.status == 200X{ 


var msg = xmlhttp.response Text; /获取 xmlhttp 对 象 返 回 的 文本 值 
if(msg != '1){ 

alert(' 删 除 失败 '); /如 果 为 1， 说 明 删 除 失败 
jelse{ 

alert(' 删 除 成 功 '); 

location=('?page=shopcar ); // 否 则 ， 说 明 删 除 成 功 ， 刷 新 购物 车 
} 


10.10.6 ”保存 购物 车 


国 ” 保 存 购物 车 使 用 的 数据 表 : tb_commo、tb_user 

当 用 户 希 望 保存 商品 更 改 后 的 商品 数量 时 ， 可 以 单 击 “ 继 续 购 物 ” 按 钮 ， 将 触发 onClick 事件 调用 
conshop() 函 数 保存 数据 ， 该 函数 有 一 个 参数 ， 就 是 当前 表单 的 名 称 。 在 conshop() 函 数 内 ， 根 据 复 选 杠 
和 商品 数量 文本 域 ， 生 成 两 个 数组 fst 和 snd， 分 别 保存 商品 id 和 商品 数量 。 

这 里 要 注意 ,两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那么 商品 1 的 数量 就 
要 保存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 
传 信息 作出 相应 的 判断 。conshop(O) 函 数 的 代码 如 下 : 

倒 程 38 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\shopcarjs 


/* 更 改 商 品 数量 */ 

function conshop(form){ 
varn_pre = 'cnum'; // 商 品 数 量 文本 域 前 缀 
var lang = form.chk.length; // 获 取 复 选 框 的 数量 


/如 果 只 有 一 个 复 选 框 ， 那 么 将 商品 id 和 商品 数量 直接 保存 到 变量 中 */ 
if(lang == undefinedX{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 


/和 否则， 将 商品 id 和 对 应 的 商品 数量 保存 到 两 个 相应 的 数组 中 */ 


jelse{ 
var fst= new Array(); // 商 品 id 数组 
var snd = new Array(); // 商 品 数量 数组 


/循环 获取 复 选 框 的 value 值 和 商品 数量 文本 框 的 value 值 */ 
for(vari= 0;i < lang; i++){ 
varnm =n_pre+itoString(); 
var stmp = document.getElementByld(nm).value; 
/* 对 商品 数量 文本 框 判断 ， 不 允许 为 空 ， 而 且 不 允许 为 非 数 字 输 入 %/ 
if(stmp =="||isNaN(stmp)){ 
alert(' 不 允许 为 空 ， 必 须 为 数字 '); 
document.getElementByld(nm).select(); 
return false; 
} 
sndli] = stmp; 


@ 


第 10 章 电子 商务 网 站 ( Apache+PHP+ADODB+Smarty+Ajax+MySQL 实现 ) 


var ftmp = form.chk[i].value; 
fst[i] = ftmp; 


} 


全 url = "changecarphp?fst='+fst+'&snd='+snd; // 生 成 ur 
xmlhttp.open("GET",url,true); /| 调用 xmlhttp 对 象 
xmlhttp.onreadystatechange = updatecar; // 调 用 updatecar() 函 数 
xmlhttp.send(null); 


} 

/对 xmlhttp 对 象 的 返回 值 进行 处 理 */ 

function updatecar(){ 
if(xmlhttp.readyState == 4){ 


} 


var msg = xmlhttp.responseText; 


if(msg == "1'X{ 

location='index.php'; // 如 果 操 作成 功 ， 返 回首 页 
Jelse{ 

alert(' 操 作 失 败 '); // 否 则 ， 提 示 错 误 
} 


在 conshop() 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ,该 页 将 商品 id 和 商品 数量 进行 重新 排列 ， 
并 保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
倒 程 39 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\changecar.php 


<?php 
include_once 'system/system.inc.php'; /| 连接 数据 库 
$sql = 'select id,shopping from tb_user where id = '.(int)$_SESSION['id]; // 生 成 SQL 查询 语句 
Sfst = $_GET[fst]; // 获 取 商 品 id 
$snd =$_GET[snd]; // 获 取 商 品 数量 
S$reback = '0'; /设置 返回 值 ， 默 认为 0 
S$rst = $conn->execute($sql); /| 执行 查询 语句 ,返回 查询 结果 集 
$changecar = array(); 1/ 创建 更 新 数组 


if($fst != "and $snd != "}{ 


/使 用 逗号 “,” 将 商品 id 和 商品 数量 分 隔 ， 并 分 别 保存 到 两 个 临时 数组 中 */ 
$farr = explode(",', $fst); 
$sarr = explode(',', $snd); 
/临时 数组 纪 
$upcar = array(); 
/使 用 for 循环 ， 将 商品 id 和 商品 数量 一 一 对 应 ， 重 新 组 合 */ 
for($i = 0; $i < count($farr); $i++X{ 
S$upcar[$i] = $farr[$i].,".$sarr[$i]; 


} 
/如 果 数 组 大 于 1， 说 明 有 多 个 商品 ， 使 用 “@” 将 不 同 商品 分 隔 开 */ 
if(count($farr) > 1){ 
$changecar['shopping'] = implode('@', $upcar); 
}else{ 
如 果 只 有 一 个 商品 ， 则 直接 保存 到 更 新 数组 中 */ 


$changecar['shopping] = $upcar[0]; 


} 
/生成 更 新 语句 
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S$UpdateSql = $conn->GetUpdateSQL($rst$changecartrue); 


人 * 更 新 数据 库 */ 
if(false == $conn->execute($UpdateSql))\{ 
S$reback = '2'; 
Jelse{ 
S$reback = "1'; 
} 
echo $reback; 


?> 


10.11 收银 台 模块 设计 


10.11.1 ”收银 台 模 块 概述 


当 用 户 停止 浏览 商品 准备 结账 时 , 可 以 单 击 购物 车 页 面 中 的 “去 收银 台 ” 按 钮 , 该 按钮 将 触发 onClick 
事件 调用 formset0 函 数 显 示 订 单 页 面 ， 当 用 户 提交 订单 后 ， 系 统 将 订单 保存 到 数据 表 tb_form 中 , 同时 
清空 购物 车 ， 并 显示 订单 信息 提醒 用 户 记录 订单 号 。 当 货款 发 出 
后 ， 还 可 以 对 订单 进行 查询 。 收 银 台 页 面 的 运行 结果 如 图 10.24 


0 1 中 人 全 -Mrocokt Intemet Explorer FEj| 


所 示 。 [ts tt 
ne 


本 节 所 涉及 的 页 面 有 显示 订单 (formset0) 函 数 )、 填 写 订 单 
(settle.php、settle.html)、 处 理 订单 (settle_chk.php)、 反 馈 订单 
(forminfo.php、forminfo.html》 和 查询 订单 5 部 分 。 


10.11.2 ”收银 台 模块 技术 分 析 


在 收银 台 模 块 中 , 通过 ADODB 中 的 方法 完成 订单 信息 的 添加 和 数据 的 更 新 操作 。 在 ADODB 中 ， 
数据 的 添加 和 更 新 可 以 采用 两 种 方法 : 第 一 种 直接 定义 INSERT 或 者 UPDATE 语句 ， 通 过 execute() 函 
数 来 执行 添加 或 者 更 新 的 操作 ;第 二 种 使 用 ADODB 提供 的 函数 来 完成 。 


1. GetlnsertSQL() 函 数 

GetInsertSQLO 函 数 向 数据 库 中 添加 新 的 记录 ， 其 语法 如 下 : 

GetInsertSQL(&$rs, $arrFields) 

参数 $rs 设置 要 添加 记录 的 结果 集 ， 参数 SarrFields 设置 新 记录 的 字段 值 及 字段 名 称 。 
2. GetUpdateSQL() 函 数 

GetUpdateSQLO 函 数 更 新 数据 库 中 指定 的 记录 ， 其 语法 如 下 : 


GetUpdateSQL(&$rs, $arrFields, $forceUpdate=false) 


参数 $rs 指定 要 更 新 的 结果 集 ; 参数 SarrFields 指定 更 新 的 字段 和 内 容 ， 参数 $forceUpdate 控制 更 新 
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操作 是 否 执行 ， 如 果 $forceUpdate 为 tme， 无 论 $arFields 和 Srs 的 值 是 否 相 等 都 将 更 新 。 


10.11.3 ”显示 订单 


国 ” 显 示 订 单 使 用 的 数据 表 : tb_commo、tb_user 


订单 信息 提交 页 面 的 输出 由 formset0 函 数 决定 ， 它 将 商品 信息 进行 整理 ， 然 后 通过 open() 方 法 打 
开 settle.php 页 来 显示 订单 , 并 将 整理 后 的 商品 信息 传递 到 settle.php 文件 中 。 formset0 函 数 的 代码 如 下 : 


倒 性 40 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\js\shopcarjs 


function formset(formX{ 
var uid = form.uid.value; // 获 取 订 单 提交 人 的 名 称 
var n_pre ='cnum'; /商品 数量 文本 前 缀 

var lang = form.chk.length; // 复 选 框 个 数 


让 如 果 复 选 框 只 有 一 个 */ 
if(lang == undefinedX{ 


var fst = form.chk.value; /商品 id 
var snd = form.cnum0.value; /商品 数量 
让 当 有 多 个 复 选 框 时 */ 
jelse{ 


var fst= new Array(); 
var snd = new Array(); 
/使 用 for 循环 ， 获 取 商 品 数量 和 商品 id*/ 
for(vari= 0;i< lang; i++)}{ 
varnm = n_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)){ 
alert(' 不 允许 为 空 ， 必 须 为 数字 '); 
document.getElementByld(nm).select(); 
return false; 
} 
snd[i] = stmp; 
var ftmp = form.chkli].value; 
fst[] = ftmp; 


} 
/使 用 open() 方 法 打开 settle.php 页 ， 并 将 当前 用 户 会 员 名 称 、 商 品 id 和 商品 数量 的 值 传 过 去 */ 
open('settle.php?uid="+uid+'&fst="+fst+'&snd="+snd,'_blank',"width=420 height=220',false); 


NC 


” 因为 open() 方 法 使 用 了 _blank 参数 来 打开 一 个 新 的 页 面 ，session 值 传 不 过 去 ， 所 以 这 里 


使 用 隐藏 域 来 传递 用 户 名 称 。 


_ 国 ) 
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10.11.4 填写 订单 


国 填写 订单 使 用 的 数据 表 : tb_commo、tb_user、tb_form 


settle.php 直接 将 接收 的 值 传 给 settle.html 模板 , 并 载 入 settle.html 模板 。settle.php 页 面 的 代码 如 下 : 


倒 程 41 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\settle.php 


<?php 


?> 


include_once 'system/system.inc.php’; 
Sfst = $_GET[fst]; 

$snd = $_GET[snd']; 

S$uid =$_GET[uid]; 
$smarty->assign('title',' 结 账 '); 
$smarty->assign('fst', $fst); 
$smarty->assign('snd',$snd); 
$smarty->assign('uid',$uid); 
$smarty->display(settle.html); 


// 载 入 Smarty 配置 文件 
/获取 GET 属性 值 


/将 标题 、 数 组 等 数据 传递 给 模板 


// 显 示 settle.html 模板 文件 


settle.html 模板 显示 一 个 表单 ， 这 个 表单 的 内 容 需 要 用 户 来 填写 ， 包 括 收 货 人 、 联 系 电话 等 信息 。 
而 从 PHP 页 传 过 来 的 儿 个 变量 则 被 保存 到 隐藏 域 以 传递 到 处 理 页 ， 在 表单 中 将 数据 提交 到 settle_ 
chk.php 处 理 页 。 


10.11.5 ”处理 订单 


国 。 处理 订 单 使 用 的 数据 表 : tb_commo、tb_user、tb_form 


处 理 页 settle_chk.php 获取 表单 中 提交 的 数据 ,根据 用 户 提交 的 商品 信息 ,重新 查找 数据 表 tb_commo， 
并 从 数据 表 中 提取 商品 信息 ， 保 存 到 数组 中 ， 然 后 处 理 页 将 数组 作为 一 条 记录 添加 到 表 tb_form 内 。 
数据 添加 成 功 的 同时 , 处 理 页 会 根据 uid 找到 该 用 户 , 将 shopping 字段 清空 , 最 后 调用 forminfo.php 


页 来 显示 新 添加 的 订单 信息 。settle_chk.php 页 的 代码 如 下 : 
倒 程 42 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\settle_chk.php 


<?php 
/* 获 取 表单 值 ， 并 存储 到 添加 数组 中 */ 


include_once 'system/system.inc.php’; 

$sql = "select * from tb_form where id = -1"; 
$rst = $conn->execute($sql); 

$addform = array(); 

$addform[vendee] = $_POST[uid]; 
$addform[commo id] = $_POST[fst]; 
$addform[commo_num]= $_POST[snd]; 


/* 根 据 商品 id 和 商品 数量 ， 获 取 所 需 数 据 */ 


$addform[formid] = time() 
S$tmpid = explode(,,$addform[commo id]); 


$tmpnm = explode(',',$addform['commo_num"); 


$number = count($tmpid); 
广 当 购买 商品 种 类 大 于 1 时 */ 
if($number >1){ 


/提交 订单 用 户 
// 购 买 商品 id 
// 购 买 商品 数量 


// 订 单 号 

// 将 商品 id 存 为 数组 
// 将 商品 数量 存 为 数组 
// 获 取 商 品 数组 长 度 
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S$tmpna = array(); /商品 名 称 数组 
$tmpvp = array(); /商品 单价 数组 
S$tmpfd = array(); /商品 折扣 数组 
S$tmptt = 0; /商品 总 额 数 组 


/使 用 for 循环 获取 各 种 商品 的 信息 */ 
for($i = 0; $i < $number; $i++){ 
/* 生 成 查询 语句 ， 执 行 SQL 语句 ， 最 后 返回 查询 结果 集 */ 
Stmpsql = "select name,v_price,fold from tb_commo where id = ".$tmpid[$i]; 
Stmprst = $conn->execute($tmpsql); 
让 将 每 种 商品 的 信息 放 到 对 应 的 数组 中 */ 
Stmpna[$i] = $tmprst->fields['name'; 
Stmpvp[$i] = $tmprst->fields[v_price']; 
S$tmpfd[$i] = $tmprst->fields['fold"]; 
$tmptt += $tmprst->fields['v_price’] * $tmpnm[$i]; 


} 
/* 将 获取 的 信息 数组 分 别 生成 长 字符 串 ， 数 组 值 之 间 使 用 豆 号 “,” 分 隔 */ 


$addform['commo_name'] = implode(,,$tmpna); /商品 名 称 
$addform['agoprice'] = implode(,,$tmpvp); /商品 价格 
$addform['fold'] = implode(,,$tmpfd); // 商 品 折扣 
$addform['total'] = $tmptt; /商品 总 额 


/* 当 购买 一 种 商品 时 沁 
}else if($number == 1X{ 
$tmpsql = "select name,v_price,fold from tb_commo where id = ".$tmpid[0]; 
$tmprst = $conn->execute($tmpsql); 
$addform['commo_name'] = $tmprst->fields['name']; 
$addform['agoprice'] = $tmprst->fields[v_price]; 
$addform[fold'] = $tmprst->fields[fold']; 
$addform['total] = $tmprst->fields[v_price] * $tmpnm[0]; 


让 没有 商品 时 */ 
}else{ 

echo 'error'; 

exit(); 
} 
/将 表单 的 其 他 信息 存 入 添加 数组 
$addform[taker] = $_POST[taken]; /路 货 人 
$addform['code'] = $_POST[code]; 1/ 邮编 
$addform['tel] = $_POST['tel]; /联系 电话 
$addform[address] = $_POST[address'; /地 址 
$addform[del_method]= $_POST[del']; // 送 货 方式 
$addform[pay_method] = $_POST[pay]; /付款 方式 
$addform['state] ='0'; /订单 状态 ，0 为 未 处 理 
$lnsertSQL = $conn->GetInsertSQL (S$rst,$addform); /| 生成 SQL 添加 语句 


if(false == $conn->execute($InsertSQL))}{ 
* 如 果 购 买 失败 ， 返 回 上 层 */ 
echo "<script>alert(' 购 买 失 败 ');history.back;</script>"; 
jelse{ 
如 果 成 功 ， 删 除 购物 车 信息 ， 并 显示 订单 信息 */ 
$updsql = "select * from tb_user where name =".$_POST[uid].”"; 
$updrst = $conn->execute($updsql); 
$arr = array(); 
$arr[consume'] = $addform['total"]; 
$arr['shopping] = "; 
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$UpdateSQL = $conn->GetUpdateSQL($updrst,$arr); 
$conn->execute($UpdateSQL); 
echo "<script>top.opener.location.reload();</script>"; 
echo "<script>open('forminfo.php?fid=$fid',"_blank',"width=600 height=450',false);</script>”; 
echo “<script>window.close(); </script>"; 
; 
eg 


由 于 篇 幅 所 限 ， 有 关 反 馈 订 单 和 查询 订单 的 内 容 这 里 不 再 讲解 ， 请 读者 参考 本 书 光盘 中 的 源 代码 。 
10.12 后 台 首 页 设计 


10.12.1 后 台 首 页 概述 


后 台 管 理 系统 是 网 站 管理 员 对 商品 、 会 员 及 公告 等 信息 进行 统一 管理 的 场所 ， 本 系统 的 后 台 主 要 
包括 以 下 功能 。 
类 别管 理 模块 : 主要 包括 对 商品 类 别 的 添加 、 修 改 及 删除 操作 。 
商品 管理 模块 : 主要 包括 对 商品 的 添加 、 修 改 、 删 除 及 订单 处 理 。 
用 户 管理 模块 : 主要 包括 管理 员 管理 和 会 员 管理 。 其 中 ,管理 员 管 理 是 实现 对 管理 员 的 添加 、 
删除 和 修改 功能 ; 会 员 管理 则 包括 删除 和 冻结 功能 。 
公告 管理 模块 : 主要 包括 公告 的 添加 及 删除 操作 。 
链接 管理 模块 : 主要 包括 添加 、 修 改 和 删除 友情 链接 。 
后 台 首 页 的 运行 结果 如 图 10.25 所 示 。 
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要 者 订单 EC) oaae| Ee es “| 良和 条 中 ”” 平 部 ”| 未 处理 ”| 这 替 | ”法 未 处理 
人 于 Ni 


图 10.25 后 台 首页 的 运行 结果 


10.12.2 后台 首页 技术 分 析 


对 后 台 首页 的 设计 和 前 台 首 页 不 同 ， 其 使 用 的 是 框架 布局 。 框 架 布 局 的 特点 是 : 可 以 将 容器 窗口 
划分 为 若干 个 子 窗 口 ， 每 个 子 窗 口 可 以 分 别 显示 不 同 的 网 页 ， 网 页 之 间 为 相互 独立 的 ， 没 有 直接 的 关 
联 ， 又 由 一 个 网 页 将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 者 的 浏览 器 中 。 框 架 布局 的 好 
处 是 : 每 次 浏览 者 发 出 对 页 面 的 请 求 时 ， 只 下 载 发 生变 化 的 框架 页 面 ， 其 他 子 页 面 保持 不 变 。 下 面 来 
具体 看 一 下 框架 布局 的 使 用 格式 及 属性 。 


@ 
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1. 框架 布局 格式 
框架 布局 的 格式 很 简单 ， 只 要 几 行 代码 即 可 ， 常 用 的 格式 如 下 : 


<html> 
<head> 


</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 
<body> 


</body> 
</noframes> 
</html> 


其 中 <frameset> 和 <frame> 标 签 是 框架 集 标 记 ， 


而 <noframes> 标 签 是 为 了 防止 浏览 器 不 支持 框架 而 


实行 的 一 种 补救 措施 。 如 果 浏 览 器 不 支持 框架 集 ， 就 会 执行 <noframes> 标 记 中 的 内 容 ， 让 用 户 能 够 正 


2. 框架 集 属性 


框架 集 包含 各 个 框架 的 信息 ， 通 过 <frameset> 标 记 来 定义 。 框 架 是 按照 行 和 列 来 组 织 的 ， 可 以 使 用 
<framesef> 标 记 的 属性 对 框架 的 结构 进行 设置 。 框 架 集 的 常用 属性 值 、 说 明和 应 用 举例 如 表 10.1 所 示 。 


表 10.1 框架 集 的 常用 属性 


参 数 说 了 明 举例 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， | <frameset cols="25%,100,*" > 
COLS 取 值 有 3 种 形式 : 像素 、 百 分 比 〈%) 和 | <frame></frame> 
相对 尺寸 (*) </frameset> 
<frameset rows="25%,100,*" > 
BOWE 在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， <frame> 
取 值 和 COLS 类 似 ， 也 是 3 种 形式 <frame> 
</frameset> 
指定 框架 周围 是 否 显 示 边 框 , 取 值 为 1( 显 <framset cols="25%,*" cols="*" frameborder="0"> 
TRAMEBORDER | 示 边框 ,默认 值 ) 或 0 (不 显示 边框 
</frameset> 
<framset cols="25%,*" cols="*" framespacing="1"> 
FRAMESPACING | 指定 框架 之 间 的 间隔， 以 像素 为 单位 。 默 Re 
认 是 无 间隔 的 
</frameset> 
<framset cols="25%,*" cols="*" frameborder="1" 
指定 边框 的 宽度 , frameborder 属性 为 1 时 | border="5"> 
BORDER ji 


该 属性 才 有 效 


</frameset> 
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3. 框架 属性 


使 用 <frame> 标 记 可 以 设置 框架 的 属性 ， 包 括 框架 的 名 称 ， 框 架 是 否 包含 滚动 条 以 及 在 框架 中 显示 
的 网 页 等 。<frame> 标 记 的 常用 属性 及 其 说 明 如 表 10.2 所 示 。 


表 10.2 框架 属性 
参数 说 了 明 
NAME 指定 框架 的 名 称 
SRC 指定 在 框架 中 显示 的 网 页 文件 (包括 HTML、PHP、JSP 等 网 页 文件 ) 
FRAMEBODER | 。 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框 ， 为 默认 ) 或 0 (不 显示 边框 ) 
NORESIZE | ”可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 


SCROLLING 指定 框架 是 否 包含 滚动 条 。 属 性 可 以 是 yes (有 )、no (没有 ) 和 auto( 自 由 ) 


10.12.3 后台 首页 实现 过 程 


国 后台 首页 使 用 的 数据 表 : tb_ commo、tb_admin、tb_public、tb_links、tb_class 
(1) 定义 框架 页 面 main.html， 包 含 3 个 文件 : top.html、leftphp 和 default.php。main.html 页 的 代 
码 如 下 : 
倒 程 43 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\admin\main.html 


<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<link rel="stytlesheet" href="css/style.css" /> 
<title> 鹏 逊 电子 商务 后 台 管 理 系统 </title> 
</head> 
<!-- 使 用 frame 框架 布局 --> 
<frameset rows="126,” cols="*" frameborder="no" border="0" framespacing="0"> 
<!-- 头 部 框架 ， 命 名 为 topFrame， 载 入 文件 为 top.html--> 
<frame src="top.php" name="topFrame" scrolling="No" noresize="noresize" id="topFrame" title="topFrame" /> 
<frameset rows="*" cols="210,*" framespacing="0" frameborder="no" border="0"> 
<!-- 左 侧 框 架 ， 名 字 为 leftFrame， 载 入 文件 为 leff.php--> 
<frame src="leftphp”name="leftFrame" frameborder="0" scrolling="auto" noresize="noresize" id="leftFrame”" 
title="leftFrame" /> 
<!- ”中 部 框架 ， 名 字 为 mainFrame， 载 入 文件 为 default.php 一 > 
<frame src="default.php" name="mainFrame" id="mainFrame" title="mainFrame" /> 
</frameset> 
</frameset> 
<noframes><body></body></noframes> 
</html> 


(2)left.php 页 是 一 个 树 形 菜单 ,应 用 DIV+JavaScript+CSS 来 实现 的 .首先 介绍 div 标签 ,在 left.html 
模板 文件 中 ， 其 关键 代码 如 下 : 
倒 性 44 ”代码 位 置 : 光盘 \TM\10\ShoppingSystem\admin\left.html 


<!-- 载 入 css 样式 和 JavaScript 脚本 -> 
<link href="css/left.css" rel="stylesheet" type="text/css" /> 
<script language="javascript" src="js/left.js"></script> 


@ 
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<! 一 类 别管 理 菜单 -> 

<div id="type" align="center" onclick="javascript:change(one,type);"> 类 别管 理 </div> 

<!-- 子 菜单 --> 

<div id="one" style="display: "> 

<div id="addtype" align="center><a href="addtype.php" target="mainFrame" id="menu"> 添 加 类 别 </a></div> 
<div id="showtype" align="center"><a href="showtype.php" target="mainFrame" id="menu"> 查 看 类 别 </a></div> 
</div> 

<div id="hidediv" align="center></div> 

<!-- 商 品 管理 菜单 -> 

<div id="commo" align="center onclick="javascript:change(two,type);"> 类 别管 理 </div> 

<div id="two"style="display:none"> 

<!-- 商 品 管理 子 菜单 --> 


</div> 


4 
说 日 
i 除了 加 粗 的 id 名 称 和 JavaScript 事件 不 同 外 ， 其 他 菜单 的 结构 完全 相同 ， 此 时 只 需 修改 超 
链接 即 可 。 


和 注意 从 了 第 一 个 类 列 某 单 的 于 某 音 splay 样 区 为 室外 其 他 由 个 于 某 单 的 加 y 料 坟 部 为 Aisplay- 


none;”。 


该 页 面 在 Dreamweaver 中 的 效果 如 图 10.26 所 示 。 
和 - 关 别 黄河 


= 


图 10.26 div 树 形 菜单 


因为 其 他 子 菜单 的 样式 为 display=none， 所 以 只 有 “类 别管 理 ” 子 菜单 是 可 见 的 ， 下 面 为 它 添加 
JavaScript 事件 。leftjs 脚本 文件 的 代码 如 下 : 
倒 程 45 ”代码 位 置 ， 光盘 \TM\10\ShoppingSystem\admin\js\left.js 
/函数 功能 : 改变 子 菜单 样式 ， 更 换 当 前 菜单 的 背景 图 片 
*nu: 为 子 菜单 div 标签 的 id 号 
* |x: 为 调用 change() 函 数 所 在 的 标签 id 
过 
function change(nu,lx){ 
/* 如 果 当 前 子 莱 单 的 样式 为 none， 则 显示 子 菜单 */ 
if(nu.style.display == "none")}{ 
nu.style.display = ™"; 
Ix.style.background="url(images/main_openroot.gif)"; 
/和 否则， 隐藏 当前 菜单 的 子 菜单 六 
jelse{ 
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nu.style.display = "none"; 
Ix.style.background="url(images/main_closeroot.gif)"; 
» 


最 后 在 left.css 中 设置 div 的 长 、 宽 等 一 些 默认 参数 。 至 此 ， 一 个 简单 而 又 实用 的 树 形 菜单 就 完成 了 。 
对 于 后 台 的 大 部 分 模块 来 说 ， 其 功能 实现 的 方法 和 开发 步骤 在 前 台 的 模块 设计 中 基本 都 已 经 介绍 
过 。 由 于 篇 幅 所 限 ， 这 里 不 再 对 后 台 管 理 模块 进行 详细 的 讲解 。 


10.13 ”开发 技巧 与 难点 分 析 


在 本 系统 开发 和 后 期 测试 的 过 程 中 ， 开 发 人 员 遇 到 了 各 种 各 样 的 疑难 问题 。 这 里 找 出 一 些 常见 的 、 
容易 被 忽略 的 问题 加 以 讲解 ， 希 望 能 够 为 初学 者 和 新 手提 供 一 些 帮 助 ， 在 开发 程序 时 少 走 一 些 弯路 。 


10.13.1 解决 Ajax 的 乱码 问题 


问题 描述 : 当 使 用 Ajax 传递 数据 时 ， 要 么 在 数据 处 理 页 中 数据 不 能 被 正确 处 理 ， 要 么 输出 返回 值 
时 显示 的 是 一 堆 无 法 识别 的 乱码 。 

解决 方法 : 这 是 因为 PHP 在 传递 数据 时 使 用 的 编码 默认 为 UTF-8， 这 就 造成 了 非 英 文字 符 不 能 正 
确 传递 的 情况 。 解 决 方法 如 下 : 

在 所 有 的 PHP 页 中 都 输入 代码 “Header("Content-Type:text/html;charset=gb2312");”， 这 样 所 有 的 页 
面 即 可 正确 显示 。 


10.13.2 ”使 用 JavaScript 脚本 获取 、 输 出 标签 内 容 


问题 描述 : 获取、 更 改 表单 元 素 值 和 特定 标签 内 容 。 

解决 方法 : 使 用 JavaScript 脚本 获取 页 面 内 容 的 方式 主要 有 两 种 ， 第 一 种 是 通过 表单 获取 表单 元 素 
的 value 值 。 格 式 为 : 表单 名 称 .元 素 名 .value。 该 方式 只 能 获取 表单 中 的 元 素 值 ， 对 于 其 他 标签 元 素 则 
无 能 为 力 。 而 第 二 种 方式 可 以 通过 id 名 来 获取 页 面 中 任意 标签 的 内 容 。 格 式 为 “ document. 
getElementById('id). value; ”或 “document.getElementById (“id’).innerText;”。 

使 用 第 二 种 方式 时 要 注意 ,标签 的 id 名 必须 存在 且 唯 一 ， 和 否则 就 会 出 现 错误 。 为 标签 内 容 赋值 时 ， 
则 使 用 如 下 格式 : 


id.innerHTML =' 要 显示 的 内 容 '; 


10.13.3 ”禁用 页 面 缓存 


问题 描述 : 使 用 Ajax 技术 可 以 防止 页 面 刷新 , 但 有 时 也 会 产生 新 的 问题 。 例 如 ,在 会 员 管理 页 面 ， 
如 果 连 续 地 冻结 和 解冻 会 员 ， 那 么 超过 3 次 后 ， 该 功能 将 失效 ， 因 为 在 一 定时 间 内 ， 如 果 做 相同 的 操 
作 ， 那 么 xmlhttprequest 对 象 会 执行 缓存 中 的 信息 ， 从 而 造成 操作 失败 。 

解决 办 法 : 使 用 header() 函 数 将 缓存 关闭 。 将 代码 “header("CACHE-CONTROL:NO-CACHE'");” 
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添加 到 xmlhttprequest 对 象 所 调用 的 处 理 页 的 顶部 即 可 。 
10.13.4 在 新 窗口 中 使 用 session 


问题 描述 : 使 用 JavaScript 文件 的 open0) 方 法 打开 新 窗口 时 ， 原 浏览 器 中 的 session 值 不 会 被 传递 
到 新 窗口 中 ， 从 而 造成 数据 查询 失败 。 
解决 方法 : 将 session 值 另 存 到 隐藏 域 或 随 着 url 一 起 传递 到 新 窗口 。 代 码 如 下 : 


<! 一 在 模板 页 中 ， 将 session 值 赋 给 隐藏 域 -> 
<input id="uid" name="uid" type="hidden" value="{$smarty.session.id}"> 


上 * 在 JavaScript 脚本 中 ， 获 取 到 隐藏 域 value 值 */ 
function getlnput(){ 
Var uid = document.getElementByld('uid').value; 
/* 将 获取 的 value 值 通过 url 传 给 新 页 面 */ 
open("operator.php?uid="+uid,"_blank',",false); 


10.13.5 判断 上 传 文件 格式 


问题 描述 : 添加 商品 时 可 以 上 传 商品 的 图 片 ， 但 有 时 可 能 会 误 传 非 图 片 格式 的 文件 ， 从 而 导致 上 
传 图 片 失败 。 

解决 方法 : 这 里 就 自 定义 一 个 函数 来 判断 上 传 文件 的 后 级 ， 创 建 自 定义 函数 fpostfix0， 函 数 的 代 
人 码 如 下 : 

六 

* 判 断 文 件 后 缀 

*$f_type: 允许 文件 的 后 缀 类 型 (数组 ) 

*$f_upfiles: 上 传 文件 名 

学 

function f_postfix($f_type, $f_upfilesX{ 


S$is_pass = false; 

$tmp_upfiles = split("\.",$f_upfiles); /使 用 split() 函 数 分 隔 文件 

S$tmp_num = count($tmp_upfiles); // 查 找 文件 后 缀 

if(in_array(strtolower($tmp_upfiles[$tmp_num - 1]), $f_type)) // 判 断后 缀 是 否 在 允许 列表 内 
S$is_pass = $tmp_upfiles[$tmp_num - 1]; 1/ 如果 是 ， 则 将 后 缀 名 赋 给 变量 

return $is_pass; // 返 回 变量 


10.13.6 ”设置 服务 器 的 时 间 
问题 描述 ， 如 果 没 有 对 PHP 的 时 区 进行 设置 ， 那 么 使 用 日 期 、 时 间 函 数 获取 的 将 是 英国 伦敦 本 地 


时 间 ( 即 零 时 区 的 时 间 )。 例 如 ， 以 东 八 区 为 例 ， 如 果 当 地 使 用 的 是 北京 时 间 ， 那 么 如 果 没 有 对 PHP 
的 时 区 进行 设置 ， 获 取 的 时 间 将 比 当地 的 北京 时 间 少 8 个 小 时 。 
_- 国 
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解决 方案 : 要 获取 本 地 当前 的 时 间 必 须 更 改 PHP 语言 中 的 时 区 设置 。 更 改 PHP 语言 中 的 时 区 设置 
有 以 下 两 种 方法 : 
(1) 在 php.ini 文件 中 ， 定 位 到 [date] 下 的 “;date.timezone =” 选 项 ， 去 掉 前 面 的 分 号 ， 并 设置 它 
的 值 为 当地 所 在 时 区 使 用 的 时 间 。 修 改 内 容 如 图 10.27 所 示 。 
例如 ， 如 果 当地 所 在 时 区 为 东 八 区 ， 那 么 就 可 以 设置 “date. Es 
timezone=” 的 值 为 PRC、Asia/Hong Kong、Asia/Shanghai (上 ee i | 
海 ) 或 者 Asia/Urumqi (乌鲁木齐 ) 等 。 这 些 都 是 东 八 区 的 时 间 。 
设置 完成 后 ， 保 存 文件 ， 重 新 启动 Apache 服务 器 。 
(2) 在 应 用 程序 中 ， 在 日 期 、 时 间 函 数 之 前 使 用 
date_default_timezone_set() 函 数 就 可 以 完成 对 时 区 的 设置 。date_default_timezone_set() 函 数 的 语法 如 下 : 


ldate Functions | 
; http://php.net/date.tinezone 


图 10.27 设置 PHP 的 时 


区 


date_default_timezone_set(timezone); 


参数 timezone 为 PHP 可 识别 的 时 区 名 称 ， 如 果 时 区 名 称 PHP 无 法 识别 ， 则 系统 采用 UTC 时 区 。 
例如 ， 设 置 北京 时 间 可 以 使 用 的 时 区 包括 PRC (中 华人 民 共 和 国 )、Asia/Chongqing (重庆 )、 
Asia/Shanghai (上 海 ) 或 者 Asia/Urumqi (乌鲁木齐 )， 这 几 个 时 区 名 称 是 等 效 的 。 


10.14 Smarty 类 库 技 术 专 题 


Smarty 是 一 个 使 用 PHP 编写 的 PHP 模板 引擎 ， 是 目前 业界 最 著名 、 功 能 最 强大 的 一 种 PHP 模板 
引擎 。 它 将 一 个 应 用 程序 分 成 两 部 分 : 视图 和 人 逻辑 控制 ， 也 就 是 将 UI (用 户 界面 ) 和 PHP code (PHP 
代码 ) 分 离 。 这 样 ， 程 序 员 在 修改 程序 时 不 会 影响 到 页 面 设计 ， 而 美工 在 重新 设计 或 修改 页 面 时 也 不 
会 影响 程序 逻辑 。Smarty 模板 引擎 的 运行 流程 如 图 10.28 所 示 。 


如 果 是 第 一 
模板 文件 次 或 模板 已 
(页面 设计 ) 经 改变 ， 重 

新 生成 文件 


i | 解析 模板 
这 到 柄 析 文 件 | 加 村 机 
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图 10.28 Smarty 模板 引擎 的 运行 流程 
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10.14.1 Smarty 类 库 下 载 、 安 装 


PHP 没有 内 置 Smarty 模板 类 ， 需 要 单独 下 载 和 配置 。 并 且 Smarty 要 求 服务 器 上 的 PHP 版 本 最 低 
为 4.0.6。 用 户 可 以 去 http://smarty.net/download.php 下 载 最 新 的 Smarty 压缩 包 。 本 书 中 使 用 的 版 本 是 
Smarty-2.6.23， 下 载 地 址 如 图 10.29 所 示 。 


Latest Stable Release 
smarty 2625 CC Lp) Way 23rd, 2003 


Previous Releases 


io bonod Meriomdonl Priore ih tt si CE 


10.29 Smarty 下 载 


(1) 将 下 载 的 Smarty 压缩 包 解 压 ， 解 压 后 的 文件 夹 如 图 10.30 所 示 。 其 中 有 一 个 libs 目录 ， 这 里 
包含 Smarty 类 库 的 核心 文件 :Smarty.class.php、Smarty_Compiler.class.php、Config_File.class.php 和 
debug.tp14 个 文件 ， 还 有 internals 和 plugins 两 个 目录 ， 如 图 10.31 所 示 。 

(2) Smarty 类 库 的 载 入 ， 即 将 libs 文件 夹 复制 到 服务 器 根 目录 下 ， 并 为 其 重新 命名 。 一 般 该 目录 的 
名 称 为 smarty、class 等 ， 这 里 将 libs 文件 夹 重新 命名 为 Smarty， 并 且 复 制 到 章 文件 夹 10\ShoppingSystem\ 
system\smarty 下 ， 如 图 10.32 所 示 。 
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图 10.30 Smarty-2.6.23 的 压缩 包 图 10.31 Smarty-2.6.23 的 libs 文件 图 10.32 ”Smarty 类 库 的 存储 位 置 
10.14.2 ”Smarty 类 库 的 配置 


Smarty 模板 的 配置 步骤 如 下 : 
(1) 确定 Smarty 目录 的 位 置 。 因 为 Smarty 类 库 是 通用 的 ， 每 一 个 项 目 都 可 能 会 使 用 到 它 。 所 以 
将 Smarty 存储 在 根 目录 下 ,在 本 项 目 中 将 Smarty 类 库 文 件 存储 在 10\ShoppingSystem\system\ 文 件 夹 下 ， 
命名 为 Smarty。 
(2) 新 建 ttmplates、templates c、configs 和 cache4 个 目录 。 其 中 目录 templates 存储 项 目的 模板 


人 
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文件 ; 目录 templates_c 存储 项 目的 编译 文件 ; 目录 configs 存储 项 目的 配置 文件 ; 目录 cache 存储 项 目 
的 缓存 文件 。 这 4 个 目录 具体 放置 在 什么 位 置 没 有 严格 的 规定 ， 只 要 设置 的 路 径 正确 ， 放 在 哪里 都 可 
以 。 在 本 项 目 中 将 创建 的 4 个 文件 夹 存储 在 10\ShoppingSystem\system\ 文 件 夹 下 。 

(3) 创建 配置 文件 。 如 果 要 应 用 Smarty 模板 ， 那 么 就 一 定 要 包含 Smarty 类 库 和 相关 信息 。 配 置 
信息 可 以 存储 在 一 个 单独 的 文件 中 ， 在 需要 时 通过 include 语句 包含 即 可 。 配 置 文件 config.php 的 代码 
如 下 : 


<?php 

定义 服务 器 的 绝对 路 径 */ 
define(BASE_PATH',$_SERVER[DOCUMENT_ROOT]); 
/定义 Smarty 模板 的 绝对 路 径 */ 
define('SMARTY_PATH',\mi\14\Smarty\'); 

* 加 载 Smarty 类 库 文 件 */ 

require BASE_PATH.SMARTY_PATH.'Smarty.class.php'; 
/实例 化 一 个 Smarty 对 象 */ 

$smarty = new Smarty; 


"定义 各 个 目录 的 路 径 */ 

$smarty->template_dir = "./'; /定义 模板 文件 的 存储 位 置 
$smarty->compile_dir = './Smarty/templates_c/'; // 定 义 编译 文件 的 存储 位 置 
$smarty->config_dir = './Smarty/configs/"; // 定 义 配置 文件 的 存储 位 置 
$smarty->cache_dir = './Smarty/cache/; /定义 缓存 文件 的 存储 位 置 
?> 


上 述 配置 文件 的 参数 说 明 如 下 。 
BASE_PATH: 指定 服务 器 的 绝对 路 径 。 
SMARTY_PATH: 指定 Smarty 模板 的 绝对 路 径 。 
require(): 加 载 Smarty 类 库 文件 Smarty.class.php。 
$smarty: 实例 化 Smarty 对 象 。 
S$smarty->template_dir: 定义 模板 目录 存储 位 置 。 
$smarty-> compile_dir: 定义 编译 目录 存储 位 置 。 
$smarty-> config_dir: 定义 配置 文件 存储 位 置 。 
$smarty-> cache_dir: 定义 模板 缓存 目录 。 
以 上 是 最 基本 的 配置 方法 ， 而 在 本 项 目 中 使 用 的 配置 方法 更 加 实用 、 灵 活 ， 在 10.14.3 节 将 对 其 进 
行 详 细 讲解 。 
10.14.3 本 项 目 中 Smarty 类 库 的 配置 方法 
本 项 目 中 将 Smarty 配置 方法 封装 到 类 中 ， 并 存储 到 system.smarty.inc.php 文件 中 。 
首先 ， 包 含 Smarty 类 文件 Smarty.class.php。 然 后 ， 定 义 SmartyProject 类 ， 继 承 Smarty 父 类 。 最 


后 ， 定 义 SmartyProject() 方 法 ， 设 置 Smarty 中 模板 文件 (templates )、 编 译文 件 (templates c)、 配 置 文 
件 (configs 〉 和 缓存 文件 (cache) 的 存储 位 置 。system.smarty.inc.php 的 代码 如 下 : 


加 


加 回回 轿 图 图 加 


<?php 
require("../Smarty/Smarty.class.php"); ll 调用 Smarty 文件 
class SmartyProject extends Smarty{ // 定 义 类 ， 继 承 Smarty 父 类 


@ 
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function SmartyProject()f /定义 方法 ， 配 置 Smarty 模板 
S$this->template_dir = "./"; // 指 定 模板 文件 存储 在 根 目录 下 
S$this->compile_dir = "./system/Smarty/templates_c/"; 。 ”// 指 定编 译文 件 存储 位 置 
Sthis->config_dir = "./system/Smarty/configs/"; 
S$this->cache_dir = "./system/Smarty/cache/"; 
} 


} 
es 


既然 已 经 将 Smarty 的 配置 方法 存储 到 一 个 类 中 ， 那 么 就 需要 对 类 进行 实例 化 ， 根 据 返 回 的 对 象 名 
称 调用 Smarty 中 的 方法 。 类 的 实例 化 操作 在 system.inc.php 文件 中 完成 ， 其 返回 对 象 名 为 Ssmarty， 代 
人 码 如 下 : 


<?php 
require("system.smarty.inc.php"); /1 调用 类 文件 
$smarty=new SmartyProject(); /| 执行 类 的 实例 化 操作 
让 


通过 此 方法 配置 Smarty 模板 的 好 处 是 : 无 论 将 程序 复制 到 哪个 服务 器 下 执行 ， 都 不 需要 更 改 服务 
器 或 者 Smarty 文件 的 绝对 路 径 ， 程 序 就 可 以 直接 运行 。 


10.15 本 章 总 结 


本 章 使 用 Smarty、ADODB、Ajax 等 目前 的 主流 技术 ， 实 现 一 个 电子 商务 平台 从 系统 分 析 到 最 后 
发 布 的 全 过 程 。 希 望 读者 能 通过 这 个 项 目 实例 ， 把 前 面 所 学 到 的 各 种 技术 消化 吸收 、 融 会 贯通 ， 并 能 
够 学 以 臻 用、 举一反三 。 


